mirror of https://github.com/procxx/kepka.git
Save item views in App::*Item() variables.
Also remove App::contextItem. Also use owning pointers for history context menus.
This commit is contained in:
parent
8060cb7426
commit
062b0b2165
|
@ -89,11 +89,10 @@ namespace {
|
|||
using SentData = QMap<uint64, QPair<PeerId, QString>>;
|
||||
SentData sentData;
|
||||
|
||||
HistoryItem *hoveredItem = nullptr,
|
||||
HistoryView::Message *hoveredItem = nullptr,
|
||||
*pressedItem = nullptr,
|
||||
*hoveredLinkItem = nullptr,
|
||||
*pressedLinkItem = nullptr,
|
||||
*contextItem = nullptr,
|
||||
*mousedItem = nullptr;
|
||||
|
||||
QPixmap *emoji = nullptr, *emojiLarge = nullptr;
|
||||
|
@ -934,7 +933,7 @@ namespace {
|
|||
: nullptr);
|
||||
existing->setViewsCount(m.has_views() ? m.vviews.v : -1);
|
||||
existing->indexAsNewItem();
|
||||
if (!existing->detached()) {
|
||||
if (existing->mainView()) {
|
||||
App::checkSavedGif(existing);
|
||||
return true;
|
||||
}
|
||||
|
@ -1836,23 +1835,20 @@ namespace {
|
|||
}
|
||||
}
|
||||
|
||||
void historyItemDetached(HistoryItem *item) {
|
||||
if (::hoveredItem == item) {
|
||||
void messageViewDestroyed(not_null<HistoryView::Message*> view) {
|
||||
if (::hoveredItem == view) {
|
||||
hoveredItem(nullptr);
|
||||
}
|
||||
if (::pressedItem == item) {
|
||||
if (::pressedItem == view) {
|
||||
pressedItem(nullptr);
|
||||
}
|
||||
if (::hoveredLinkItem == item) {
|
||||
if (::hoveredLinkItem == view) {
|
||||
hoveredLinkItem(nullptr);
|
||||
}
|
||||
if (::pressedLinkItem == item) {
|
||||
if (::pressedLinkItem == view) {
|
||||
pressedLinkItem(nullptr);
|
||||
}
|
||||
if (::contextItem == item) {
|
||||
contextItem(nullptr);
|
||||
}
|
||||
if (::mousedItem == item) {
|
||||
if (::mousedItem == view) {
|
||||
mousedItem(nullptr);
|
||||
}
|
||||
}
|
||||
|
@ -1867,7 +1863,6 @@ namespace {
|
|||
data->erase(i);
|
||||
}
|
||||
}
|
||||
historyItemDetached(item);
|
||||
auto j = ::dependentItems.find(item);
|
||||
if (j != ::dependentItems.cend()) {
|
||||
DependentItemsSet items;
|
||||
|
@ -1903,13 +1898,13 @@ namespace {
|
|||
|
||||
QVector<HistoryItem*> toDelete;
|
||||
for_const (auto item, msgsData) {
|
||||
if (item->detached()) {
|
||||
if (!item->mainView()) {
|
||||
toDelete.push_back(item);
|
||||
}
|
||||
}
|
||||
for_const (auto &chMsgsData, channelMsgsData) {
|
||||
for_const (auto item, chMsgsData) {
|
||||
if (item->detached()) {
|
||||
if (!item->mainView()) {
|
||||
toDelete.push_back(item);
|
||||
}
|
||||
}
|
||||
|
@ -2187,51 +2182,43 @@ namespace {
|
|||
clearAllImages();
|
||||
}
|
||||
|
||||
void hoveredItem(HistoryItem *item) {
|
||||
void hoveredItem(HistoryView::Message *item) {
|
||||
::hoveredItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *hoveredItem() {
|
||||
HistoryView::Message *hoveredItem() {
|
||||
return ::hoveredItem;
|
||||
}
|
||||
|
||||
void pressedItem(HistoryItem *item) {
|
||||
void pressedItem(HistoryView::Message *item) {
|
||||
::pressedItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *pressedItem() {
|
||||
HistoryView::Message *pressedItem() {
|
||||
return ::pressedItem;
|
||||
}
|
||||
|
||||
void hoveredLinkItem(HistoryItem *item) {
|
||||
void hoveredLinkItem(HistoryView::Message *item) {
|
||||
::hoveredLinkItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *hoveredLinkItem() {
|
||||
HistoryView::Message *hoveredLinkItem() {
|
||||
return ::hoveredLinkItem;
|
||||
}
|
||||
|
||||
void pressedLinkItem(HistoryItem *item) {
|
||||
void pressedLinkItem(HistoryView::Message *item) {
|
||||
::pressedLinkItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *pressedLinkItem() {
|
||||
HistoryView::Message *pressedLinkItem() {
|
||||
return ::pressedLinkItem;
|
||||
}
|
||||
|
||||
void contextItem(HistoryItem *item) {
|
||||
::contextItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *contextItem() {
|
||||
return ::contextItem;
|
||||
}
|
||||
|
||||
void mousedItem(HistoryItem *item) {
|
||||
void mousedItem(HistoryView::Message *item) {
|
||||
::mousedItem = item;
|
||||
}
|
||||
|
||||
HistoryItem *mousedItem() {
|
||||
HistoryView::Message *mousedItem() {
|
||||
return ::mousedItem;
|
||||
}
|
||||
|
||||
|
@ -2240,7 +2227,6 @@ namespace {
|
|||
pressedItem(nullptr);
|
||||
hoveredLinkItem(nullptr);
|
||||
pressedLinkItem(nullptr);
|
||||
contextItem(nullptr);
|
||||
mousedItem(nullptr);
|
||||
}
|
||||
|
||||
|
|
|
@ -15,6 +15,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
class Messenger;
|
||||
class MainWindow;
|
||||
class MainWidget;
|
||||
class LocationCoords;
|
||||
struct LocationData;
|
||||
|
||||
namespace HistoryView {
|
||||
class Message;
|
||||
} // namespace HistoryView
|
||||
|
||||
using HistoryItemsMap = OrderedSet<HistoryItem*>;
|
||||
using PhotoItems = QHash<PhotoData*, HistoryItemsMap>;
|
||||
|
@ -27,9 +33,6 @@ using GifItems = QHash<Media::Clip::Reader*, HistoryItem*>;
|
|||
using PhotosData = QHash<PhotoId, PhotoData*>;
|
||||
using DocumentsData = QHash<DocumentId, DocumentData*>;
|
||||
|
||||
class LocationCoords;
|
||||
struct LocationData;
|
||||
|
||||
namespace App {
|
||||
MainWindow *wnd();
|
||||
MainWidget *main();
|
||||
|
@ -190,13 +193,13 @@ namespace App {
|
|||
return histItemById(msgId.channel, msgId.msg);
|
||||
}
|
||||
void historyRegItem(HistoryItem *item);
|
||||
void historyItemDetached(HistoryItem *item);
|
||||
void historyUnregItem(HistoryItem *item);
|
||||
void historyUpdateDependent(HistoryItem *item);
|
||||
void historyClearMsgs();
|
||||
void historyClearItems();
|
||||
void historyRegDependency(HistoryItem *dependent, HistoryItem *dependency);
|
||||
void historyUnregDependency(HistoryItem *dependent, HistoryItem *dependency);
|
||||
void messageViewDestroyed(not_null<HistoryView::Message*> view);
|
||||
|
||||
void historyRegRandom(uint64 randomId, const FullMsgId &itemId);
|
||||
void historyUnregRandom(uint64 randomId);
|
||||
|
@ -205,18 +208,16 @@ namespace App {
|
|||
void historyUnregSentData(uint64 randomId);
|
||||
void histSentDataByItem(uint64 randomId, PeerId &peerId, QString &text);
|
||||
|
||||
void hoveredItem(HistoryItem *item);
|
||||
HistoryItem *hoveredItem();
|
||||
void pressedItem(HistoryItem *item);
|
||||
HistoryItem *pressedItem();
|
||||
void hoveredLinkItem(HistoryItem *item);
|
||||
HistoryItem *hoveredLinkItem();
|
||||
void pressedLinkItem(HistoryItem *item);
|
||||
HistoryItem *pressedLinkItem();
|
||||
void contextItem(HistoryItem *item);
|
||||
HistoryItem *contextItem();
|
||||
void mousedItem(HistoryItem *item);
|
||||
HistoryItem *mousedItem();
|
||||
void hoveredItem(HistoryView::Message *item);
|
||||
HistoryView::Message *hoveredItem();
|
||||
void pressedItem(HistoryView::Message *item);
|
||||
HistoryView::Message *pressedItem();
|
||||
void hoveredLinkItem(HistoryView::Message *item);
|
||||
HistoryView::Message *hoveredLinkItem();
|
||||
void pressedLinkItem(HistoryView::Message *item);
|
||||
HistoryView::Message *pressedLinkItem();
|
||||
void mousedItem(HistoryView::Message *item);
|
||||
HistoryView::Message *mousedItem();
|
||||
void clearMousedItems();
|
||||
|
||||
const style::font &monofont();
|
||||
|
|
|
@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "lang/lang_keys.h"
|
||||
#include "messenger.h"
|
||||
#include "platform/platform_specific.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "boxes/confirm_box.h"
|
||||
#include "base/qthelp_regex.h"
|
||||
#include "base/qthelp_url.h"
|
||||
|
@ -241,10 +242,11 @@ void BotCommandClickHandler::onClick(Qt::MouseButton button) const {
|
|||
}
|
||||
|
||||
if (auto peer = Ui::getPeerForMouseAction()) { // old way
|
||||
UserData *bot = peer->isUser() ? peer->asUser() : nullptr;
|
||||
if (auto item = App::hoveredLinkItem()) {
|
||||
if (!bot) {
|
||||
bot = item->fromOriginal()->asUser(); // may return nullptr
|
||||
auto bot = peer->isUser() ? peer->asUser() : nullptr;
|
||||
if (!bot) {
|
||||
if (const auto view = App::hoveredLinkItem()) {
|
||||
// may return nullptr
|
||||
bot = view->data()->fromOriginal()->asUser();
|
||||
}
|
||||
}
|
||||
Ui::showPeerHistory(peer, ShowAtTheEndMsgId);
|
||||
|
|
|
@ -527,7 +527,6 @@ inline int ceilclamp(float64 value, int32 step, int32 lowest, int32 highest) {
|
|||
|
||||
enum ForwardWhatMessages {
|
||||
ForwardSelectedMessages,
|
||||
ForwardContextMessage,
|
||||
ForwardPressedMessage,
|
||||
ForwardPressedLinkMessage
|
||||
};
|
||||
|
|
|
@ -336,7 +336,9 @@ void DocumentSaveClickHandler::doSave(
|
|||
bool forceSavingAs) {
|
||||
if (!data->date) return;
|
||||
|
||||
auto filepath = data->filepath(DocumentData::FilePathResolveSaveFromDataSilent, forceSavingAs);
|
||||
auto filepath = data->filepath(
|
||||
DocumentData::FilePathResolveSaveFromDataSilent,
|
||||
forceSavingAs);
|
||||
if (!filepath.isEmpty() && !forceSavingAs) {
|
||||
File::OpenWith(filepath, QCursor::pos());
|
||||
} else {
|
||||
|
@ -345,9 +347,7 @@ void DocumentSaveClickHandler::doSave(
|
|||
auto filename = filepath.isEmpty() ? QString() : fileinfo.fileName();
|
||||
auto newfname = documentSaveFilename(data, forceSavingAs, filename, filedir);
|
||||
if (!newfname.isEmpty()) {
|
||||
auto action = (filename.isEmpty() || forceSavingAs) ? ActionOnLoadNone : ActionOnLoadOpenWith;
|
||||
auto actionMsgId = App::hoveredLinkItem() ? App::hoveredLinkItem()->fullId() : (App::contextItem() ? App::contextItem()->fullId() : FullMsgId());
|
||||
data->save(newfname, action, actionMsgId);
|
||||
data->save(newfname, ActionOnLoadNone, FullMsgId());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -362,8 +362,7 @@ void DocumentCancelClickHandler::onClickImpl() const {
|
|||
|
||||
if (data->uploading()) {
|
||||
if (const auto item = App::histItemById(context())) {
|
||||
App::contextItem(item);
|
||||
App::main()->cancelUploadLayer();
|
||||
App::main()->cancelUploadLayer(item);
|
||||
}
|
||||
} else {
|
||||
data->cancel();
|
||||
|
|
|
@ -137,8 +137,7 @@ void PhotoCancelClickHandler::onClickImpl() const {
|
|||
|
||||
if (data->uploading()) {
|
||||
if (const auto item = App::histItemById(context())) {
|
||||
App::contextItem(item);
|
||||
App::main()->cancelUploadLayer();
|
||||
App::main()->cancelUploadLayer(item);
|
||||
}
|
||||
} else {
|
||||
data->cancel();
|
||||
|
|
|
@ -44,9 +44,5 @@ void MessageCursor::applyTo(QTextEdit *edit) {
|
|||
HistoryItem *FileClickHandler::getActionItem() const {
|
||||
return context()
|
||||
? App::histItemById(context())
|
||||
: App::hoveredLinkItem()
|
||||
? App::hoveredLinkItem()
|
||||
: App::contextItem()
|
||||
? App::contextItem()
|
||||
: nullptr;
|
||||
}
|
||||
|
|
|
@ -418,13 +418,14 @@ void InnerWidget::updateEmptyText() {
|
|||
|
||||
QString InnerWidget::tooltipText() const {
|
||||
if (_mouseCursorState == HistoryInDateCursorState && _mouseAction == MouseAction::None) {
|
||||
if (const auto item = App::hoveredItem()) {
|
||||
auto dateText = item->date.toString(QLocale::system().dateTimeFormat(QLocale::LongFormat));
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
auto dateText = view->data()->date.toString(
|
||||
QLocale::system().dateTimeFormat(QLocale::LongFormat));
|
||||
return dateText;
|
||||
}
|
||||
} else if (_mouseCursorState == HistoryInForwardedCursorState && _mouseAction == MouseAction::None) {
|
||||
if (const auto item = App::hoveredItem()) {
|
||||
if (const auto forwarded = item->Get<HistoryMessageForwarded>()) {
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
if (const auto forwarded = view->data()->Get<HistoryMessageForwarded>()) {
|
||||
return forwarded->text.originalText(AllTextSelection, ExpandLinksNone);
|
||||
}
|
||||
}
|
||||
|
@ -808,10 +809,6 @@ void InnerWidget::contextMenuEvent(QContextMenuEvent *e) {
|
|||
}
|
||||
|
||||
void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
||||
if (_menu) {
|
||||
_menu->deleteLater();
|
||||
_menu = 0;
|
||||
}
|
||||
if (e->reason() == QContextMenuEvent::Mouse) {
|
||||
mouseActionUpdate(e->globalPos());
|
||||
}
|
||||
|
@ -828,10 +825,10 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
if (App::mousedItem() && App::mousedItem() == App::hoveredItem()) {
|
||||
auto mousePos = mapPointToItem(
|
||||
mapFromGlobal(_mousePosition),
|
||||
viewForItem(App::mousedItem()));
|
||||
App::mousedItem());
|
||||
HistoryStateRequest request;
|
||||
request.flags |= Text::StateRequest::Flag::LookupSymbol;
|
||||
auto dragState = App::mousedItem()->getState(mousePos, request);
|
||||
auto dragState = App::mousedItem()->data()->getState(mousePos, request);
|
||||
if (dragState.cursor == HistoryInTextCursorState && dragState.symbol >= selFrom && dragState.symbol < selTo) {
|
||||
isUponSelected = 1;
|
||||
}
|
||||
|
@ -841,10 +838,12 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
isUponSelected = hasSelected;
|
||||
}
|
||||
|
||||
_menu = new Ui::PopupMenu(nullptr);
|
||||
_menu = base::make_unique_q<Ui::PopupMenu>(nullptr);
|
||||
|
||||
_contextMenuLink = ClickHandler::getActive();
|
||||
auto item = App::hoveredItem() ? App::hoveredItem() : App::hoveredLinkItem();
|
||||
auto view = App::hoveredItem()
|
||||
? App::hoveredItem()
|
||||
: App::hoveredLinkItem();
|
||||
auto lnkPhoto = dynamic_cast<PhotoClickHandler*>(_contextMenuLink.get());
|
||||
auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get());
|
||||
auto lnkPeer = dynamic_cast<PeerClickHandler*>(_contextMenuLink.get());
|
||||
|
@ -853,41 +852,52 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
auto lnkIsAudio = lnkDocument ? lnkDocument->document()->isAudioFile() : false;
|
||||
if (lnkPhoto || lnkDocument) {
|
||||
if (isUponSelected > 0) {
|
||||
_menu->addAction(lang(lng_context_copy_selected), [this] { copySelectedText(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_selected), [=] {
|
||||
copySelectedText();
|
||||
})->setEnabled(true);
|
||||
}
|
||||
if (lnkPhoto) {
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, photo = lnkPhoto->photo()] {
|
||||
const auto photo = lnkPhoto->photo();
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [=] {
|
||||
savePhotoToFile(photo);
|
||||
}))->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_image), [this, photo = lnkPhoto->photo()] {
|
||||
_menu->addAction(lang(lng_context_copy_image), [=] {
|
||||
copyContextImage(photo);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
auto document = lnkDocument->document();
|
||||
if (document->loading()) {
|
||||
_menu->addAction(lang(lng_context_cancel_download), [this] { cancelContextDownload(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_cancel_download), [=] {
|
||||
cancelContextDownload(document);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
if (document->loaded() && document->isGifv()) {
|
||||
if (!cAutoPlayGif()) {
|
||||
_menu->addAction(lang(lng_context_open_gif), [this] { openContextGif(); })->setEnabled(true);
|
||||
const auto itemId = view
|
||||
? view->data()->fullId()
|
||||
: FullMsgId();
|
||||
_menu->addAction(lang(lng_context_open_gif), [=] {
|
||||
openContextGif(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (!document->filepath(DocumentData::FilePathResolveChecked).isEmpty()) {
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [this] { showContextInFolder(); })->setEnabled(true);
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [=] {
|
||||
showContextInFolder(document);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
_menu->addAction(lang(lnkIsVideo ? lng_context_save_video : (lnkIsVoice ? lng_context_save_audio : (lnkIsAudio ? lng_context_save_audio_file : lng_context_save_file))), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
}))->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (App::hoveredLinkItem()) {
|
||||
App::contextItem(App::hoveredLinkItem());
|
||||
}
|
||||
} else if (lnkPeer) { // suggest to block
|
||||
if (auto user = lnkPeer->peer()->asUser()) {
|
||||
suggestRestrictUser(user);
|
||||
}
|
||||
} else { // maybe cursor on some text history item?
|
||||
const auto item = view ? view->data().get() : nullptr;
|
||||
const auto itemId = item ? item->fullId() : FullMsgId();
|
||||
bool canDelete = item && item->canDelete() && (item->id > 0 || !item->serviceMsg());
|
||||
bool canForward = item && item->canForward();
|
||||
|
||||
|
@ -903,9 +913,11 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
media = static_cast<HistoryWebPage*>(media)->attach();
|
||||
}
|
||||
if (media->type() == MediaTypeSticker) {
|
||||
if (auto document = media->getDocument()) {
|
||||
if (const auto document = media->getDocument()) {
|
||||
if (document->sticker() && document->sticker()->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
_menu->addAction(lang(document->sticker()->setInstalled() ? lng_context_pack_info : lng_context_pack_add), [this] { showStickerPackInfo(); });
|
||||
_menu->addAction(lang(document->sticker()->setInstalled() ? lng_context_pack_info : lng_context_pack_add), [=] {
|
||||
showStickerPackInfo(document);
|
||||
});
|
||||
}
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
|
@ -914,15 +926,21 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
} else if (media->type() == MediaTypeGif && !_contextMenuLink) {
|
||||
if (auto document = media->getDocument()) {
|
||||
if (document->loading()) {
|
||||
_menu->addAction(lang(lng_context_cancel_download), [this] { cancelContextDownload(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_cancel_download), [=] {
|
||||
cancelContextDownload(document);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
if (document->isGifv()) {
|
||||
if (!cAutoPlayGif()) {
|
||||
_menu->addAction(lang(lng_context_open_gif), [this] { openContextGif(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_open_gif), [=] {
|
||||
openContextGif(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (!document->filepath(DocumentData::FilePathResolveChecked).isEmpty()) {
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [this] { showContextInFolder(); })->setEnabled(true);
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [=] {
|
||||
showContextInFolder(document);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
_menu->addAction(lang(lng_context_save_file), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
|
@ -932,7 +950,9 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
}
|
||||
}
|
||||
if (msg && !_contextMenuLink && (!msg->emptyText() || mediaHasTextForCopy)) {
|
||||
_menu->addAction(lang(lng_context_copy_text), [this] { copyContextText(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_text), [=] {
|
||||
copyContextText(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -941,17 +961,11 @@ void InnerWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
if (!linkCopyToClipboardText.isEmpty()) {
|
||||
_menu->addAction(linkCopyToClipboardText, [this] { copyContextUrl(); })->setEnabled(true);
|
||||
}
|
||||
App::contextItem(item);
|
||||
}
|
||||
|
||||
if (_menu->actions().isEmpty()) {
|
||||
delete base::take(_menu);
|
||||
_menu = nullptr;
|
||||
} else {
|
||||
connect(_menu, &QObject::destroyed, this, [this](QObject *object) {
|
||||
if (_menu == object) {
|
||||
_menu = nullptr;
|
||||
}
|
||||
});
|
||||
_menu->popup(e->globalPos());
|
||||
e->accept();
|
||||
}
|
||||
|
@ -961,11 +975,15 @@ void InnerWidget::savePhotoToFile(PhotoData *photo) {
|
|||
if (!photo || !photo->date || !photo->loaded()) return;
|
||||
|
||||
auto filter = qsl("JPEG Image (*.jpg);;") + FileDialog::AllFilesFilter();
|
||||
FileDialog::GetWritePath(lang(lng_save_photo), filter, filedialogDefaultName(qsl("photo"), qsl(".jpg")), base::lambda_guarded(this, [this, photo](const QString &result) {
|
||||
if (!result.isEmpty()) {
|
||||
photo->full->pix().toImage().save(result, "JPG");
|
||||
}
|
||||
}));
|
||||
FileDialog::GetWritePath(
|
||||
lang(lng_save_photo),
|
||||
filter,
|
||||
filedialogDefaultName(qsl("photo"), qsl(".jpg")),
|
||||
base::lambda_guarded(this, [=](const QString &result) {
|
||||
if (!result.isEmpty()) {
|
||||
photo->full->pix().toImage().save(result, "JPG");
|
||||
}
|
||||
}));
|
||||
}
|
||||
|
||||
void InnerWidget::saveDocumentToFile(DocumentData *document) {
|
||||
|
@ -988,50 +1006,28 @@ void InnerWidget::copyContextUrl() {
|
|||
}
|
||||
}
|
||||
|
||||
void InnerWidget::showStickerPackInfo() {
|
||||
if (!App::contextItem()) return;
|
||||
|
||||
if (auto media = App::contextItem()->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
if (auto sticker = doc->sticker()) {
|
||||
if (sticker->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
App::main()->stickersBox(sticker->set);
|
||||
}
|
||||
}
|
||||
void InnerWidget::showStickerPackInfo(not_null<DocumentData*> document) {
|
||||
if (auto sticker = document->sticker()) {
|
||||
if (sticker->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
App::main()->stickersBox(sticker->set);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void InnerWidget::cancelContextDownload() {
|
||||
if (auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get())) {
|
||||
lnkDocument->document()->cancel();
|
||||
} else if (auto item = App::contextItem()) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
doc->cancel();
|
||||
}
|
||||
}
|
||||
}
|
||||
void InnerWidget::cancelContextDownload(not_null<DocumentData*> document) {
|
||||
document->cancel();
|
||||
}
|
||||
|
||||
void InnerWidget::showContextInFolder() {
|
||||
QString filepath;
|
||||
if (auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get())) {
|
||||
filepath = lnkDocument->document()->filepath(DocumentData::FilePathResolveChecked);
|
||||
} else if (auto item = App::contextItem()) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
filepath = doc->filepath(DocumentData::FilePathResolveChecked);
|
||||
}
|
||||
}
|
||||
}
|
||||
void InnerWidget::showContextInFolder(not_null<DocumentData*> document) {
|
||||
const auto filepath = document->filepath(
|
||||
DocumentData::FilePathResolveChecked);
|
||||
if (!filepath.isEmpty()) {
|
||||
File::ShowInFolder(filepath);
|
||||
}
|
||||
}
|
||||
|
||||
void InnerWidget::openContextGif() {
|
||||
if (auto item = App::contextItem()) {
|
||||
void InnerWidget::openContextGif(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto document = media->getDocument()) {
|
||||
Messenger::Instance().showDocument(document, item);
|
||||
|
@ -1040,16 +1036,20 @@ void InnerWidget::openContextGif() {
|
|||
}
|
||||
}
|
||||
|
||||
void InnerWidget::copyContextText() {
|
||||
auto item = App::contextItem();
|
||||
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
|
||||
return;
|
||||
void InnerWidget::copyContextText(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (const auto media = item->getMedia()) {
|
||||
if (media->type() == MediaTypeSticker) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
setToClipboard(item->selectedText(FullSelection));
|
||||
}
|
||||
|
||||
setToClipboard(item->selectedText(FullSelection));
|
||||
}
|
||||
|
||||
void InnerWidget::setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode) {
|
||||
void InnerWidget::setToClipboard(
|
||||
const TextWithEntities &forClipboard,
|
||||
QClipboard::Mode mode) {
|
||||
if (auto data = MimeDataFromTextWithEntities(forClipboard)) {
|
||||
QApplication::clipboard()->setMimeData(data.release(), mode);
|
||||
}
|
||||
|
@ -1057,6 +1057,7 @@ void InnerWidget::setToClipboard(const TextWithEntities &forClipboard, QClipboar
|
|||
|
||||
void InnerWidget::suggestRestrictUser(not_null<UserData*> user) {
|
||||
Expects(_menu != nullptr);
|
||||
|
||||
if (!_channel->isMegagroup() || !_channel->canBanMembers() || _admins.empty()) {
|
||||
return;
|
||||
}
|
||||
|
@ -1065,8 +1066,8 @@ void InnerWidget::suggestRestrictUser(not_null<UserData*> user) {
|
|||
return;
|
||||
}
|
||||
}
|
||||
_menu->addAction(lang(lng_context_restrict_user), [this, user] {
|
||||
auto editRestrictions = [user, this](bool hasAdminRights, const MTPChannelBannedRights ¤tRights) {
|
||||
_menu->addAction(lang(lng_context_restrict_user), [=] {
|
||||
auto editRestrictions = [=](bool hasAdminRights, const MTPChannelBannedRights ¤tRights) {
|
||||
auto weak = QPointer<InnerWidget>(this);
|
||||
auto weakBox = std::make_shared<QPointer<EditRestrictedBox>>();
|
||||
auto box = Box<EditRestrictedBox>(_channel, user, hasAdminRights, currentRights);
|
||||
|
@ -1085,7 +1086,7 @@ void InnerWidget::suggestRestrictUser(not_null<UserData*> user) {
|
|||
if (base::contains(_admins, user)) {
|
||||
editRestrictions(true, MTP_channelBannedRights(MTP_flags(0), MTP_int(0)));
|
||||
} else {
|
||||
request(MTPchannels_GetParticipant(_channel->inputChannel, user->inputUser)).done([this, editRestrictions](const MTPchannels_ChannelParticipant &result) {
|
||||
request(MTPchannels_GetParticipant(_channel->inputChannel, user->inputUser)).done([=](const MTPchannels_ChannelParticipant &result) {
|
||||
Expects(result.type() == mtpc_channels_channelParticipant);
|
||||
|
||||
auto &participant = result.c_channels_channelParticipant();
|
||||
|
@ -1162,8 +1163,8 @@ void InnerWidget::enterEventHook(QEvent *e) {
|
|||
}
|
||||
|
||||
void InnerWidget::leaveEventHook(QEvent *e) {
|
||||
if (auto item = App::hoveredItem()) {
|
||||
repaintItem(viewForItem(item));
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
repaintItem(view);
|
||||
App::hoveredItem(nullptr);
|
||||
}
|
||||
ClickHandler::clearActive();
|
||||
|
@ -1181,13 +1182,13 @@ void InnerWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butt
|
|||
|
||||
ClickHandler::pressed();
|
||||
if (App::pressedItem() != App::hoveredItem()) {
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
repaintItem(App::pressedItem());
|
||||
App::pressedItem(App::hoveredItem());
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
repaintItem(App::pressedItem());
|
||||
}
|
||||
|
||||
_mouseAction = MouseAction::None;
|
||||
_mouseActionItem = viewForItem(App::mousedItem());
|
||||
_mouseActionItem = App::mousedItem();
|
||||
_dragStartPosition = mapPointToItem(
|
||||
mapFromGlobal(screenPos),
|
||||
_mouseActionItem);
|
||||
|
@ -1270,8 +1271,8 @@ void InnerWidget::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton but
|
|||
if (_mouseAction == MouseAction::Dragging) {
|
||||
activated = nullptr;
|
||||
}
|
||||
if (App::pressedItem()) {
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
if (const auto view = App::pressedItem()) {
|
||||
repaintItem(view);
|
||||
App::pressedItem(nullptr);
|
||||
}
|
||||
|
||||
|
@ -1318,23 +1319,25 @@ void InnerWidget::updateSelected() {
|
|||
const auto view = (from != end) ? from->get() : nullptr;
|
||||
const auto item = view ? view->data().get() : nullptr;
|
||||
if (item) {
|
||||
App::mousedItem(item);
|
||||
App::mousedItem(view);
|
||||
itemPoint = mapPointToItem(point, view);
|
||||
if (item->hasPoint(itemPoint)) {
|
||||
if (App::hoveredItem() != item) {
|
||||
repaintItem(viewForItem(App::hoveredItem()));
|
||||
App::hoveredItem(item);
|
||||
if (App::hoveredItem() != view) {
|
||||
repaintItem(App::hoveredItem());
|
||||
App::hoveredItem(view);
|
||||
repaintItem(view);
|
||||
}
|
||||
} else if (App::hoveredItem()) {
|
||||
repaintItem(viewForItem(App::hoveredItem()));
|
||||
} else if (const auto view = App::hoveredItem()) {
|
||||
repaintItem(view);
|
||||
App::hoveredItem(nullptr);
|
||||
}
|
||||
}
|
||||
|
||||
HistoryTextState dragState;
|
||||
ClickHandlerHost *lnkhost = nullptr;
|
||||
auto selectingText = (view == _mouseActionItem && item == App::hoveredItem() && _selectedItem);
|
||||
auto selectingText = _selectedItem
|
||||
&& (view == _mouseActionItem)
|
||||
&& (view == App::hoveredItem());
|
||||
if (view) {
|
||||
if (view != _mouseActionItem || (itemPoint - _dragStartPosition).manhattanLength() >= QApplication::startDragDistance()) {
|
||||
if (_mouseAction == MouseAction::PrepareDrag) {
|
||||
|
@ -1422,13 +1425,9 @@ void InnerWidget::updateSelected() {
|
|||
}
|
||||
|
||||
// Voice message seek support.
|
||||
if (auto pressedItem = App::pressedLinkItem()) {
|
||||
if (!pressedItem->detached()) {
|
||||
if (pressedItem->history() == _history) {
|
||||
auto adjustedPoint = mapPointToItem(point, viewForItem(pressedItem));
|
||||
pressedItem->updatePressed(adjustedPoint);
|
||||
}
|
||||
}
|
||||
if (const auto pressedView = App::pressedLinkItem()) {
|
||||
const auto adjustedPoint = mapPointToItem(point, pressedView);
|
||||
pressedView->data()->updatePressed(adjustedPoint);
|
||||
}
|
||||
|
||||
//if (_mouseAction == MouseAction::Selecting) {
|
||||
|
|
|
@ -121,12 +121,12 @@ private:
|
|||
void savePhotoToFile(PhotoData *photo);
|
||||
void saveDocumentToFile(DocumentData *document);
|
||||
void copyContextImage(PhotoData *photo);
|
||||
void showStickerPackInfo();
|
||||
void showStickerPackInfo(not_null<DocumentData*> document);
|
||||
void copyContextUrl();
|
||||
void cancelContextDownload();
|
||||
void showContextInFolder();
|
||||
void openContextGif();
|
||||
void copyContextText();
|
||||
void cancelContextDownload(not_null<DocumentData*> document);
|
||||
void showContextInFolder(not_null<DocumentData*> document);
|
||||
void openContextGif(FullMsgId itemId);
|
||||
void copyContextText(FullMsgId itemId);
|
||||
void copySelectedText();
|
||||
TextWithEntities getSelectedText() const;
|
||||
void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard);
|
||||
|
@ -228,8 +228,7 @@ private:
|
|||
bool _wasSelectedText = false; // was some text selected in current drag action
|
||||
Qt::CursorShape _cursor = style::cur_default;
|
||||
|
||||
// context menu
|
||||
Ui::PopupMenu *_menu = nullptr;
|
||||
base::unique_qptr<Ui::PopupMenu> _menu;
|
||||
|
||||
QPoint _trippleClickPoint;
|
||||
base::Timer _trippleClickTimer;
|
||||
|
|
|
@ -376,9 +376,9 @@ void ChannelHistory::getRangeDifference() {
|
|||
for (auto blockIndex = 0, blocksCount = int(blocks.size()); blockIndex < blocksCount; ++blockIndex) {
|
||||
const auto &block = blocks[blockIndex];
|
||||
for (auto itemIndex = 0, itemsCount = int(block->messages.size()); itemIndex < itemsCount; ++itemIndex) {
|
||||
const auto &message = block->messages[itemIndex];
|
||||
if (message->id() > 0) {
|
||||
fromId = message->id();
|
||||
const auto id = block->messages[itemIndex]->data()->id;
|
||||
if (id > 0) {
|
||||
fromId = id;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -389,9 +389,9 @@ void ChannelHistory::getRangeDifference() {
|
|||
for (auto blockIndex = blocks.size(); blockIndex > 0;) {
|
||||
const auto &block = blocks[--blockIndex];
|
||||
for (auto itemIndex = block->messages.size(); itemIndex > 0;) {
|
||||
const auto &message = block->messages[--itemIndex];
|
||||
if (message->id() > 0) {
|
||||
toId = message->id();
|
||||
const auto id = block->messages[--itemIndex]->data()->id;
|
||||
if (id > 0) {
|
||||
toId = id;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -420,11 +420,16 @@ void ChannelHistory::getRangeDifferenceNext(int32 pts) {
|
|||
}
|
||||
|
||||
HistoryJoined *ChannelHistory::insertJoinedMessage(bool unread) {
|
||||
if (_joinedMessage || !peer->asChannel()->amIn() || (peer->isMegagroup() && peer->asChannel()->mgInfo->joinedMessageFound)) {
|
||||
if (_joinedMessage
|
||||
|| !peer->asChannel()->amIn()
|
||||
|| (peer->isMegagroup()
|
||||
&& peer->asChannel()->mgInfo->joinedMessageFound)) {
|
||||
return _joinedMessage;
|
||||
}
|
||||
|
||||
auto inviter = (peer->asChannel()->inviter > 0) ? App::userLoaded(peer->asChannel()->inviter) : nullptr;
|
||||
const auto inviter = (peer->asChannel()->inviter > 0)
|
||||
? App::userLoaded(peer->asChannel()->inviter)
|
||||
: nullptr;
|
||||
if (!inviter) return nullptr;
|
||||
|
||||
MTPDmessage::Flags flags = 0;
|
||||
|
@ -491,7 +496,7 @@ void ChannelHistory::checkJoinedMessage(bool createUnread) {
|
|||
if (isEmpty()) {
|
||||
if (loadedAtTop() && loadedAtBottom()) {
|
||||
if (insertJoinedMessage(createUnread)) {
|
||||
if (!_joinedMessage->detached()) {
|
||||
if (_joinedMessage->mainView()) {
|
||||
setLastMessage(_joinedMessage);
|
||||
}
|
||||
}
|
||||
|
@ -508,7 +513,7 @@ void ChannelHistory::checkJoinedMessage(bool createUnread) {
|
|||
if (!firstDate.isNull() && !lastDate.isNull() && (firstDate <= inviteDate || loadedAtTop()) && (lastDate > inviteDate || loadedAtBottom())) {
|
||||
bool willBeLastMsg = (inviteDate >= lastDate);
|
||||
if (insertJoinedMessage(createUnread && willBeLastMsg) && willBeLastMsg) {
|
||||
if (!_joinedMessage->detached()) {
|
||||
if (_joinedMessage->mainView()) {
|
||||
setLastMessage(_joinedMessage);
|
||||
}
|
||||
}
|
||||
|
@ -545,8 +550,8 @@ void ChannelHistory::cleared(bool leaveItems) {
|
|||
_joinedMessage = nullptr;
|
||||
}
|
||||
|
||||
void ChannelHistory::messageDetached(HistoryItem *msg) {
|
||||
if (_joinedMessage == msg) {
|
||||
void ChannelHistory::messageDetached(not_null<HistoryItem*> message) {
|
||||
if (_joinedMessage == message) {
|
||||
_joinedMessage = nullptr;
|
||||
}
|
||||
}
|
||||
|
@ -565,7 +570,9 @@ History *Histories::find(const PeerId &peerId) {
|
|||
not_null<History*> Histories::findOrInsert(const PeerId &peerId) {
|
||||
auto i = map.constFind(peerId);
|
||||
if (i == map.cend()) {
|
||||
auto history = peerIsChannel(peerId) ? static_cast<History*>(new ChannelHistory(peerId)) : (new History(peerId));
|
||||
auto history = peerIsChannel(peerId)
|
||||
? static_cast<History*>(new ChannelHistory(peerId))
|
||||
: (new History(peerId));
|
||||
i = map.insert(peerId, history);
|
||||
}
|
||||
return i.value();
|
||||
|
@ -574,7 +581,9 @@ not_null<History*> Histories::findOrInsert(const PeerId &peerId) {
|
|||
not_null<History*> Histories::findOrInsert(const PeerId &peerId, int32 unreadCount, int32 maxInboxRead, int32 maxOutboxRead) {
|
||||
auto i = map.constFind(peerId);
|
||||
if (i == map.cend()) {
|
||||
auto history = peerIsChannel(peerId) ? static_cast<History*>(new ChannelHistory(peerId)) : (new History(peerId));
|
||||
auto history = peerIsChannel(peerId)
|
||||
? static_cast<History*>(new ChannelHistory(peerId))
|
||||
: (new History(peerId));
|
||||
i = map.insert(peerId, history);
|
||||
history->setUnreadCount(unreadCount);
|
||||
history->inboxReadBefore = maxInboxRead + 1;
|
||||
|
@ -634,7 +643,7 @@ void Histories::step_typings(TimeMs ms, bool timer) {
|
|||
}
|
||||
|
||||
void Histories::remove(const PeerId &peer) {
|
||||
Map::iterator i = map.find(peer);
|
||||
const auto i = map.find(peer);
|
||||
if (i != map.cend()) {
|
||||
typing.remove(i.value());
|
||||
delete i.value();
|
||||
|
@ -717,14 +726,17 @@ void Histories::checkSelfDestructItems() {
|
|||
}
|
||||
}
|
||||
|
||||
HistoryItem *History::createItem(const MTPMessage &msg, bool applyServiceAction, bool detachExistingItem) {
|
||||
HistoryItem *History::createItem(
|
||||
const MTPMessage &msg,
|
||||
bool applyServiceAction,
|
||||
bool detachExistingItem) {
|
||||
const auto msgId = idFromMessage(msg);
|
||||
if (!msgId) return nullptr;
|
||||
|
||||
auto result = App::histItemById(channelId(), msgId);
|
||||
if (result) {
|
||||
if (!result->detached() && detachExistingItem) {
|
||||
result->detach();
|
||||
if (detachExistingItem) {
|
||||
result->removeMainView();
|
||||
}
|
||||
if (msg.type() == mtpc_message) {
|
||||
const auto media = msg.c_message().has_media()
|
||||
|
@ -1053,13 +1065,18 @@ HistoryItem *History::addNewMessage(const MTPMessage &msg, NewMessageType type)
|
|||
return addNewToLastBlock(msg, type);
|
||||
}
|
||||
|
||||
HistoryItem *History::addNewToLastBlock(const MTPMessage &msg, NewMessageType type) {
|
||||
HistoryItem *History::addNewToLastBlock(
|
||||
const MTPMessage &msg,
|
||||
NewMessageType type) {
|
||||
Expects(type != NewMessageExisting);
|
||||
|
||||
const auto applyServiceAction = (type == NewMessageUnread);
|
||||
const auto detachExistingItem = (type != NewMessageLast);
|
||||
const auto item = createItem(msg, applyServiceAction, detachExistingItem);
|
||||
if (!item || !item->detached()) {
|
||||
const auto item = createItem(
|
||||
msg,
|
||||
applyServiceAction,
|
||||
detachExistingItem);
|
||||
if (!item || item->mainView()) {
|
||||
return item;
|
||||
}
|
||||
const auto result = addNewItem(item, (type == NewMessageUnread));
|
||||
|
@ -1367,14 +1384,14 @@ HistoryBlock *History::prepareBlockForAddingItem() {
|
|||
};
|
||||
|
||||
void History::addItemToBlock(not_null<HistoryItem*> item) {
|
||||
Expects(item->detached());
|
||||
Expects(!item->mainView());
|
||||
|
||||
auto block = prepareBlockForAddingItem();
|
||||
|
||||
block->messages.push_back(std::make_unique<HistoryView::Message>(
|
||||
item,
|
||||
HistoryView::Context::History));
|
||||
item->attachToBlock(block, block->messages.size() - 1);
|
||||
block->messages.back()->attachToBlock(block, block->messages.size() - 1);
|
||||
item->previousItemChanged();
|
||||
|
||||
if (isBuildingFrontBlock() && _buildingFrontBlock->expectedItemsCount > 0) {
|
||||
|
@ -1617,7 +1634,7 @@ void History::addNewerSlice(const QVector<MTPMessage> &slice) {
|
|||
|
||||
void History::checkLastMsg() {
|
||||
if (lastMsg) {
|
||||
if (!newLoaded && !lastMsg->detached()) {
|
||||
if (!newLoaded && lastMsg->mainView()) {
|
||||
newLoaded = true;
|
||||
checkAddAllToUnreadMentions();
|
||||
}
|
||||
|
@ -1840,7 +1857,7 @@ QDateTime History::adjustChatListDate() const {
|
|||
void History::countScrollState(int top) {
|
||||
countScrollTopItem(top);
|
||||
if (scrollTopItem) {
|
||||
scrollTopOffset = (top - scrollTopItem->data()->block()->y() - scrollTopItem->y());
|
||||
scrollTopOffset = (top - scrollTopItem->block()->y() - scrollTopItem->y());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1854,8 +1871,8 @@ void History::countScrollTopItem(int top) {
|
|||
auto blockIndex = 0;
|
||||
auto itemTop = 0;
|
||||
if (scrollTopItem) {
|
||||
itemIndex = scrollTopItem->data()->indexInBlock();
|
||||
blockIndex = scrollTopItem->data()->block()->indexInHistory();
|
||||
itemIndex = scrollTopItem->indexInBlock();
|
||||
blockIndex = scrollTopItem->block()->indexInHistory();
|
||||
itemTop = blocks[blockIndex]->y() + scrollTopItem->y();
|
||||
}
|
||||
if (itemTop > top) {
|
||||
|
@ -1915,7 +1932,9 @@ void History::getNextScrollTopItem(HistoryBlock *block, int32 i) {
|
|||
}
|
||||
|
||||
void History::addUnreadBar() {
|
||||
if (unreadBar || !showFrom || showFrom->detached() || !unreadCount()) return;
|
||||
if (unreadBar || !showFrom || !showFrom->mainView() || !unreadCount()) {
|
||||
return;
|
||||
}
|
||||
|
||||
int32 count = unreadCount();
|
||||
if (peer->migrateTo()) {
|
||||
|
@ -1944,14 +1963,16 @@ not_null<HistoryItem*> History::addNewInTheMiddle(
|
|||
|
||||
const auto &block = blocks[blockIndex];
|
||||
|
||||
block->messages.insert(
|
||||
const auto it = block->messages.insert(
|
||||
block->messages.begin() + itemIndex,
|
||||
std::make_unique<HistoryView::Message>(newItem, HistoryView::Context::History));
|
||||
newItem->attachToBlock(block.get(), itemIndex);
|
||||
std::make_unique<HistoryView::Message>(
|
||||
newItem,
|
||||
HistoryView::Context::History));
|
||||
(*it)->attachToBlock(block.get(), itemIndex);
|
||||
newItem->previousItemChanged();
|
||||
if (itemIndex + 1 < block->messages.size()) {
|
||||
for (auto i = itemIndex + 1, l = int(block->messages.size()); i != l; ++i) {
|
||||
block->messages[i]->data()->setIndexInBlock(i);
|
||||
block->messages[i]->setIndexInBlock(i);
|
||||
}
|
||||
block->messages[itemIndex + 1]->data()->previousItemChanged();
|
||||
} else if (blockIndex + 1 < blocks.size() && !blocks[blockIndex + 1]->messages.empty()) {
|
||||
|
@ -1969,12 +1990,12 @@ not_null<HistoryItem*> History::addNewInTheMiddle(
|
|||
}
|
||||
|
||||
HistoryItem *History::findNextItem(not_null<HistoryItem*> item) const {
|
||||
Expects(!item->detached());
|
||||
Expects(item->mainView());
|
||||
|
||||
const auto nextBlockIndex = item->block()->indexInHistory() + 1;
|
||||
const auto nextItemIndex = item->indexInBlock() + 1;
|
||||
if (nextItemIndex < int(item->block()->messages.size())) {
|
||||
return item->block()->messages[nextItemIndex]->data();
|
||||
const auto nextBlockIndex = item->mainView()->block()->indexInHistory() + 1;
|
||||
const auto nextItemIndex = item->mainView()->indexInBlock() + 1;
|
||||
if (nextItemIndex < int(item->mainView()->block()->messages.size())) {
|
||||
return item->mainView()->block()->messages[nextItemIndex]->data();
|
||||
} else if (nextBlockIndex < int(blocks.size())) {
|
||||
return blocks[nextBlockIndex]->messages.front()->data();
|
||||
}
|
||||
|
@ -1982,12 +2003,12 @@ HistoryItem *History::findNextItem(not_null<HistoryItem*> item) const {
|
|||
}
|
||||
|
||||
HistoryItem *History::findPreviousItem(not_null<HistoryItem*> item) const {
|
||||
Expects(!item->detached());
|
||||
Expects(item->mainView());
|
||||
|
||||
const auto blockIndex = item->block()->indexInHistory();
|
||||
const auto itemIndex = item->indexInBlock();
|
||||
const auto blockIndex = item->mainView()->block()->indexInHistory();
|
||||
const auto itemIndex = item->mainView()->indexInBlock();
|
||||
if (itemIndex > 0) {
|
||||
return item->block()->messages[itemIndex - 1]->data();
|
||||
return item->mainView()->block()->messages[itemIndex - 1]->data();
|
||||
} else if (blockIndex > 0) {
|
||||
return blocks[blockIndex - 1]->messages.back()->data();
|
||||
}
|
||||
|
@ -2021,7 +2042,7 @@ not_null<HistoryItem*> History::findGroupLast(
|
|||
void History::recountGroupingAround(not_null<HistoryItem*> item) {
|
||||
Expects(item->history() == this);
|
||||
|
||||
if (!item->detached() && item->groupId()) {
|
||||
if (item->mainView() && item->groupId()) {
|
||||
const auto [groupFrom, groupTill] = recountGroupingFromTill(item);
|
||||
recountGrouping(groupFrom, groupTill);
|
||||
}
|
||||
|
@ -2084,8 +2105,8 @@ auto History::recountGroupingFromTill(not_null<HistoryItem*> item)
|
|||
void History::recountGrouping(
|
||||
not_null<HistoryItem*> from,
|
||||
not_null<HistoryItem*> till) {
|
||||
Expects(!from->detached());
|
||||
Expects(!till->detached());
|
||||
Expects(from->mainView());
|
||||
Expects(till->mainView());
|
||||
|
||||
from->validateGroupId();
|
||||
auto others = std::vector<not_null<HistoryItem*>>();
|
||||
|
@ -2186,7 +2207,7 @@ bool History::isReadyFor(MsgId msgId) {
|
|||
return loadedAtBottom();
|
||||
}
|
||||
HistoryItem *item = App::histItemById(channelId(), msgId);
|
||||
return item && (item->history() == this) && !item->detached();
|
||||
return item && (item->history() == this) && item->mainView();
|
||||
}
|
||||
|
||||
void History::getReadyFor(MsgId msgId) {
|
||||
|
@ -2539,15 +2560,16 @@ void HistoryBlock::clear(bool leaveItems) {
|
|||
|
||||
if (leaveItems) {
|
||||
for (const auto &message : list) {
|
||||
message->data()->detachFast();
|
||||
message->data()->clearMainView();
|
||||
}
|
||||
}
|
||||
// #TODO feeds delete all items in history
|
||||
}
|
||||
|
||||
void HistoryBlock::removeItem(not_null<HistoryItem*> item) {
|
||||
Expects(item->block() == this);
|
||||
void HistoryBlock::remove(not_null<Message*> view) {
|
||||
Expects(view->block() == this);
|
||||
|
||||
const auto item = view->data();
|
||||
auto [groupFrom, groupTill] = _history->recountGroupingFromTill(item);
|
||||
const auto groupHistory = _history;
|
||||
const auto needGroupRecount = (groupFrom != groupTill);
|
||||
|
@ -2561,7 +2583,7 @@ void HistoryBlock::removeItem(not_null<HistoryItem*> item) {
|
|||
}
|
||||
|
||||
auto blockIndex = indexInHistory();
|
||||
auto itemIndex = item->indexInBlock();
|
||||
auto itemIndex = view->indexInBlock();
|
||||
if (_history->showFrom == item) {
|
||||
_history->getNextShowFrom(this, itemIndex);
|
||||
}
|
||||
|
@ -2575,10 +2597,10 @@ void HistoryBlock::removeItem(not_null<HistoryItem*> item) {
|
|||
_history->getNextScrollTopItem(this, itemIndex);
|
||||
}
|
||||
|
||||
item->detachFast();
|
||||
item->clearMainView();
|
||||
messages.erase(messages.begin() + itemIndex);
|
||||
for (auto i = itemIndex, l = int(messages.size()); i < l; ++i) {
|
||||
messages[i]->data()->setIndexInBlock(i);
|
||||
messages[i]->setIndexInBlock(i);
|
||||
}
|
||||
if (messages.empty()) {
|
||||
// Deletes this.
|
||||
|
|
|
@ -550,7 +550,7 @@ class ChannelHistory : public History {
|
|||
public:
|
||||
using History::History;
|
||||
|
||||
void messageDetached(HistoryItem *msg);
|
||||
void messageDetached(not_null<HistoryItem*> message);
|
||||
|
||||
void getRangeDifference();
|
||||
void getRangeDifferenceNext(int32 pts);
|
||||
|
@ -579,15 +579,17 @@ private:
|
|||
|
||||
class HistoryBlock {
|
||||
public:
|
||||
using Message = HistoryView::Message;
|
||||
|
||||
HistoryBlock(not_null<History*> history);
|
||||
HistoryBlock(const HistoryBlock &) = delete;
|
||||
HistoryBlock &operator=(const HistoryBlock &) = delete;
|
||||
~HistoryBlock();
|
||||
|
||||
std::vector<std::unique_ptr<HistoryView::Message>> messages;
|
||||
std::vector<std::unique_ptr<Message>> messages;
|
||||
|
||||
void clear(bool leaveItems = false);
|
||||
void removeItem(not_null<HistoryItem*> item);
|
||||
void remove(not_null<Message*> view);
|
||||
|
||||
int resizeGetHeight(int newWidth, bool resizeAllItems);
|
||||
int y() const {
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -28,6 +28,8 @@ class HistoryInner
|
|||
Q_OBJECT
|
||||
|
||||
public:
|
||||
using Message = HistoryView::Message;
|
||||
|
||||
HistoryInner(
|
||||
not_null<HistoryWidget*> historyWidget,
|
||||
not_null<Window::Controller*> controller,
|
||||
|
@ -40,12 +42,12 @@ public:
|
|||
TextWithEntities getSelectedText() const;
|
||||
|
||||
void touchScrollUpdated(const QPoint &screenPos);
|
||||
QPoint mapPointToItem(QPoint p, HistoryItem *item);
|
||||
|
||||
void recountHistoryGeometry();
|
||||
void updateSize();
|
||||
|
||||
void repaintItem(const HistoryItem *item);
|
||||
void repaintItem(const Message *view);
|
||||
|
||||
bool canCopySelected() const;
|
||||
bool canDeleteSelected() const;
|
||||
|
@ -68,7 +70,10 @@ public:
|
|||
int migratedTop() const;
|
||||
int historyTop() const;
|
||||
int historyDrawTop() const;
|
||||
int itemTop(const HistoryItem *item) const; // -1 if should not be visible, -2 if bad history()
|
||||
|
||||
// -1 if should not be visible, -2 if bad history()
|
||||
int itemTop(const HistoryItem *item) const;
|
||||
int itemTop(const Message *view) const;
|
||||
|
||||
void notifyIsBotChanged();
|
||||
void notifyMigrateUpdated();
|
||||
|
@ -104,14 +109,8 @@ public slots:
|
|||
void onParentGeometryChanged();
|
||||
|
||||
void copyContextUrl();
|
||||
void cancelContextDownload();
|
||||
void showContextInFolder();
|
||||
void saveContextGif();
|
||||
void openContextGif();
|
||||
void copyContextText();
|
||||
void copySelectedText();
|
||||
|
||||
void onMenuDestroy(QObject *obj);
|
||||
void onTouchSelect();
|
||||
void onTouchScrollTimer();
|
||||
|
||||
|
@ -122,7 +121,6 @@ private slots:
|
|||
private:
|
||||
class BotAbout;
|
||||
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
|
||||
using Message = HistoryView::Message;
|
||||
enum class MouseAction {
|
||||
None,
|
||||
PrepareDrag,
|
||||
|
@ -135,166 +133,6 @@ private:
|
|||
Deselect,
|
||||
Invert,
|
||||
};
|
||||
|
||||
void mouseActionStart(const QPoint &screenPos, Qt::MouseButton button);
|
||||
void mouseActionUpdate(const QPoint &screenPos);
|
||||
void mouseActionFinish(const QPoint &screenPos, Qt::MouseButton button);
|
||||
void mouseActionCancel();
|
||||
void performDrag();
|
||||
|
||||
void showContextMenu(QContextMenuEvent *e, bool showFromTouch = false);
|
||||
|
||||
void itemRemoved(not_null<const HistoryItem*> item);
|
||||
void savePhotoToFile(PhotoData *photo);
|
||||
void saveDocumentToFile(DocumentData *document);
|
||||
void copyContextImage(PhotoData *photo);
|
||||
void showStickerPackInfo(DocumentData *document);
|
||||
void toggleFavedSticker(DocumentData *document);
|
||||
|
||||
void touchResetSpeed();
|
||||
void touchUpdateSpeed();
|
||||
void touchDeaccelerate(int32 elapsed);
|
||||
|
||||
void adjustCurrent(int32 y) const;
|
||||
void adjustCurrent(int32 y, History *history) const;
|
||||
HistoryItem *prevItem(HistoryItem *item);
|
||||
HistoryItem *nextItem(HistoryItem *item);
|
||||
void updateDragSelection(HistoryItem *dragSelFrom, HistoryItem *dragSelTo, bool dragSelecting);
|
||||
TextSelection itemRenderSelection(
|
||||
not_null<Message*> view,
|
||||
int selfromy,
|
||||
int seltoy) const;
|
||||
TextSelection computeRenderSelection(
|
||||
not_null<const SelectedItems*> selected,
|
||||
not_null<HistoryItem*> item) const;
|
||||
|
||||
void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard);
|
||||
|
||||
void toggleScrollDateShown();
|
||||
void repaintScrollDateCallback();
|
||||
bool displayScrollDate() const;
|
||||
void scrollDateHide();
|
||||
void keepScrollDateForNow();
|
||||
|
||||
not_null<Window::Controller*> _controller;
|
||||
|
||||
PeerData *_peer = nullptr;
|
||||
History *_migrated = nullptr;
|
||||
History *_history = nullptr;
|
||||
int _historyPaddingTop = 0;
|
||||
|
||||
// with migrated history we perhaps do not need to display first _history message
|
||||
// (if last _migrated message and first _history message are both isGroupMigrate)
|
||||
// or at least we don't need to display first _history date (just skip it by height)
|
||||
int _historySkipHeight = 0;
|
||||
|
||||
std::unique_ptr<BotAbout> _botAbout;
|
||||
|
||||
HistoryWidget *_widget = nullptr;
|
||||
Ui::ScrollArea *_scroll = nullptr;
|
||||
mutable History *_curHistory = nullptr;
|
||||
mutable int _curBlock = 0;
|
||||
mutable int _curItem = 0;
|
||||
|
||||
bool _firstLoading = false;
|
||||
|
||||
style::cursor _cursor = style::cur_default;
|
||||
SelectedItems _selected;
|
||||
|
||||
void applyDragSelection();
|
||||
void applyDragSelection(not_null<SelectedItems*> toItems) const;
|
||||
void addSelectionRange(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<History*> history,
|
||||
int fromblock,
|
||||
int fromitem,
|
||||
int toblock,
|
||||
int toitem) const;
|
||||
bool isSelected(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
bool isSelectedAsGroup(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
bool goodForSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
int &totalCount) const;
|
||||
void addToSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
void removeFromSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
void changeSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
SelectAction action) const;
|
||||
void changeSelectionAsGroup(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
SelectAction action) const;
|
||||
void forwardItem(not_null<HistoryItem*> item);
|
||||
void forwardAsGroup(not_null<HistoryItem*> item);
|
||||
void deleteItem(not_null<HistoryItem*> item);
|
||||
void deleteAsGroup(not_null<HistoryItem*> item);
|
||||
|
||||
// Does any of the shown histories has this flag set.
|
||||
bool hasPendingResizedItems() const {
|
||||
return (_history && _history->hasPendingResizedItems()) || (_migrated && _migrated->hasPendingResizedItems());
|
||||
}
|
||||
|
||||
MouseAction _mouseAction = MouseAction::None;
|
||||
TextSelectType _mouseSelectType = TextSelectType::Letters;
|
||||
QPoint _dragStartPosition;
|
||||
QPoint _mousePosition;
|
||||
HistoryItem *_mouseActionItem = nullptr;
|
||||
HistoryItem *_dragStateItem = nullptr;
|
||||
HistoryCursorState _mouseCursorState = HistoryDefaultCursorState;
|
||||
uint16 _mouseTextSymbol = 0;
|
||||
bool _pressWasInactive = false;
|
||||
|
||||
QPoint _trippleClickPoint;
|
||||
QTimer _trippleClickTimer;
|
||||
|
||||
ClickHandlerPtr _contextMenuLink;
|
||||
|
||||
HistoryItem *_dragSelFrom = nullptr;
|
||||
HistoryItem *_dragSelTo = nullptr;
|
||||
bool _dragSelecting = false;
|
||||
bool _wasSelectedText = false; // was some text selected in current drag action
|
||||
|
||||
// scroll by touch support (at least Windows Surface tablets)
|
||||
bool _touchScroll = false;
|
||||
bool _touchSelect = false;
|
||||
bool _touchInProgress = false;
|
||||
QPoint _touchStart, _touchPrevPos, _touchPos;
|
||||
QTimer _touchSelectTimer;
|
||||
|
||||
Ui::TouchScrollState _touchScrollState = Ui::TouchScrollState::Manual;
|
||||
bool _touchPrevPosValid = false;
|
||||
bool _touchWaitingAcceleration = false;
|
||||
QPoint _touchSpeed;
|
||||
TimeMs _touchSpeedTime = 0;
|
||||
TimeMs _touchAccelerationTime = 0;
|
||||
TimeMs _touchTime = 0;
|
||||
QTimer _touchScrollTimer;
|
||||
|
||||
// context menu
|
||||
Ui::PopupMenu *_menu = nullptr;
|
||||
|
||||
// save visible area coords for painting / pressing userpics
|
||||
int _visibleAreaTop = 0;
|
||||
int _visibleAreaBottom = 0;
|
||||
|
||||
bool _scrollDateShown = false;
|
||||
Animation _scrollDateOpacity;
|
||||
SingleQueuedInvokation _scrollDateCheck;
|
||||
SingleTimer _scrollDateHideTimer;
|
||||
Message *_scrollDateLastItem = nullptr;
|
||||
int _scrollDateLastItemTop = 0;
|
||||
ClickHandlerPtr _scrollDateLink;
|
||||
|
||||
enum class EnumItemsDirection {
|
||||
TopToBottom,
|
||||
BottomToTop,
|
||||
|
@ -335,4 +173,171 @@ private:
|
|||
template <typename Method>
|
||||
void enumerateDates(Method method);
|
||||
|
||||
void mouseActionStart(const QPoint &screenPos, Qt::MouseButton button);
|
||||
void mouseActionUpdate(const QPoint &screenPos);
|
||||
void mouseActionFinish(const QPoint &screenPos, Qt::MouseButton button);
|
||||
void mouseActionCancel();
|
||||
void performDrag();
|
||||
|
||||
QPoint mapPointToItem(QPoint p, const Message *view);
|
||||
QPoint mapPointToItem(QPoint p, const HistoryItem *item);
|
||||
|
||||
void showContextMenu(QContextMenuEvent *e, bool showFromTouch = false);
|
||||
void cancelContextDownload(not_null<DocumentData*> document);
|
||||
void openContextGif(FullMsgId itemId);
|
||||
void saveContextGif(FullMsgId itemId);
|
||||
void copyContextText(FullMsgId itemId);
|
||||
void showContextInFolder(not_null<DocumentData*> document);
|
||||
void savePhotoToFile(not_null<PhotoData*> photo);
|
||||
void saveDocumentToFile(not_null<DocumentData*> document);
|
||||
void copyContextImage(not_null<PhotoData*> photo);
|
||||
void showStickerPackInfo(not_null<DocumentData*> document);
|
||||
void toggleFavedSticker(not_null<DocumentData*> document);
|
||||
|
||||
void itemRemoved(not_null<const HistoryItem*> item);
|
||||
|
||||
void touchResetSpeed();
|
||||
void touchUpdateSpeed();
|
||||
void touchDeaccelerate(int32 elapsed);
|
||||
|
||||
void adjustCurrent(int32 y) const;
|
||||
void adjustCurrent(int32 y, History *history) const;
|
||||
Message *prevItem(Message *item);
|
||||
Message *nextItem(Message *item);
|
||||
void updateDragSelection(Message *dragSelFrom, Message *dragSelTo, bool dragSelecting);
|
||||
TextSelection itemRenderSelection(
|
||||
not_null<Message*> view,
|
||||
int selfromy,
|
||||
int seltoy) const;
|
||||
TextSelection computeRenderSelection(
|
||||
not_null<const SelectedItems*> selected,
|
||||
not_null<HistoryItem*> item) const;
|
||||
|
||||
void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard);
|
||||
|
||||
void toggleScrollDateShown();
|
||||
void repaintScrollDateCallback();
|
||||
bool displayScrollDate() const;
|
||||
void scrollDateHide();
|
||||
void keepScrollDateForNow();
|
||||
|
||||
void applyDragSelection();
|
||||
void applyDragSelection(not_null<SelectedItems*> toItems) const;
|
||||
void addSelectionRange(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<History*> history,
|
||||
int fromblock,
|
||||
int fromitem,
|
||||
int toblock,
|
||||
int toitem) const;
|
||||
bool isSelected(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
bool isSelectedAsGroup(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
bool goodForSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
int &totalCount) const;
|
||||
void addToSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
void removeFromSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item) const;
|
||||
void changeSelection(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
SelectAction action) const;
|
||||
void changeSelectionAsGroup(
|
||||
not_null<SelectedItems*> toItems,
|
||||
not_null<HistoryItem*> item,
|
||||
SelectAction action) const;
|
||||
void forwardItem(FullMsgId itemId);
|
||||
void forwardAsGroup(FullMsgId itemId);
|
||||
void deleteItem(not_null<HistoryItem*> item);
|
||||
void deleteItem(FullMsgId itemId);
|
||||
void deleteAsGroup(FullMsgId itemId);
|
||||
|
||||
// Does any of the shown histories has this flag set.
|
||||
bool hasPendingResizedItems() const {
|
||||
return (_history && _history->hasPendingResizedItems()) || (_migrated && _migrated->hasPendingResizedItems());
|
||||
}
|
||||
|
||||
not_null<Window::Controller*> _controller;
|
||||
|
||||
not_null<PeerData*> _peer;
|
||||
not_null<History*> _history;
|
||||
History *_migrated = nullptr;
|
||||
int _historyPaddingTop = 0;
|
||||
|
||||
// with migrated history we perhaps do not need to display first _history message
|
||||
// (if last _migrated message and first _history message are both isGroupMigrate)
|
||||
// or at least we don't need to display first _history date (just skip it by height)
|
||||
int _historySkipHeight = 0;
|
||||
|
||||
std::unique_ptr<BotAbout> _botAbout;
|
||||
|
||||
HistoryWidget *_widget = nullptr;
|
||||
Ui::ScrollArea *_scroll = nullptr;
|
||||
mutable History *_curHistory = nullptr;
|
||||
mutable int _curBlock = 0;
|
||||
mutable int _curItem = 0;
|
||||
|
||||
bool _firstLoading = false;
|
||||
|
||||
style::cursor _cursor = style::cur_default;
|
||||
SelectedItems _selected;
|
||||
|
||||
MouseAction _mouseAction = MouseAction::None;
|
||||
TextSelectType _mouseSelectType = TextSelectType::Letters;
|
||||
QPoint _dragStartPosition;
|
||||
QPoint _mousePosition;
|
||||
HistoryItem *_mouseActionItem = nullptr;
|
||||
HistoryItem *_dragStateItem = nullptr;
|
||||
HistoryCursorState _mouseCursorState = HistoryDefaultCursorState;
|
||||
uint16 _mouseTextSymbol = 0;
|
||||
bool _pressWasInactive = false;
|
||||
|
||||
QPoint _trippleClickPoint;
|
||||
QTimer _trippleClickTimer;
|
||||
|
||||
ClickHandlerPtr _contextMenuLink;
|
||||
|
||||
Message *_dragSelFrom = nullptr;
|
||||
Message *_dragSelTo = nullptr;
|
||||
bool _dragSelecting = false;
|
||||
bool _wasSelectedText = false; // was some text selected in current drag action
|
||||
|
||||
// scroll by touch support (at least Windows Surface tablets)
|
||||
bool _touchScroll = false;
|
||||
bool _touchSelect = false;
|
||||
bool _touchInProgress = false;
|
||||
QPoint _touchStart, _touchPrevPos, _touchPos;
|
||||
QTimer _touchSelectTimer;
|
||||
|
||||
Ui::TouchScrollState _touchScrollState = Ui::TouchScrollState::Manual;
|
||||
bool _touchPrevPosValid = false;
|
||||
bool _touchWaitingAcceleration = false;
|
||||
QPoint _touchSpeed;
|
||||
TimeMs _touchSpeedTime = 0;
|
||||
TimeMs _touchAccelerationTime = 0;
|
||||
TimeMs _touchTime = 0;
|
||||
QTimer _touchScrollTimer;
|
||||
|
||||
base::unique_qptr<Ui::PopupMenu> _menu;
|
||||
|
||||
// save visible area coords for painting / pressing userpics
|
||||
int _visibleAreaTop = 0;
|
||||
int _visibleAreaBottom = 0;
|
||||
|
||||
bool _scrollDateShown = false;
|
||||
Animation _scrollDateOpacity;
|
||||
SingleQueuedInvokation _scrollDateCheck;
|
||||
SingleTimer _scrollDateHideTimer;
|
||||
Message *_scrollDateLastItem = nullptr;
|
||||
int _scrollDateLastItemTop = 0;
|
||||
ClickHandlerPtr _scrollDateLink;
|
||||
|
||||
};
|
||||
|
|
|
@ -259,7 +259,8 @@ void HistoryItem::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool activ
|
|||
markup->inlineKeyboard->clickHandlerActiveChanged(p, active);
|
||||
}
|
||||
}
|
||||
App::hoveredLinkItem(active ? this : nullptr);
|
||||
// #TODO hoveredLinkItem
|
||||
// App::hoveredLinkItem(active ? this : nullptr);
|
||||
Auth().data().requestItemRepaint(this);
|
||||
}
|
||||
|
||||
|
@ -269,7 +270,8 @@ void HistoryItem::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pres
|
|||
markup->inlineKeyboard->clickHandlerPressedChanged(p, pressed);
|
||||
}
|
||||
}
|
||||
App::pressedLinkItem(pressed ? this : nullptr);
|
||||
// #TODO hoveredLinkItem
|
||||
// App::pressedLinkItem(pressed ? this : nullptr);
|
||||
Auth().data().requestItemRepaint(this);
|
||||
}
|
||||
|
||||
|
@ -291,7 +293,7 @@ UserData *HistoryItem::viaBot() const {
|
|||
void HistoryItem::destroy() {
|
||||
const auto history = this->history();
|
||||
if (isLogEntry()) {
|
||||
Assert(detached());
|
||||
Assert(!mainView());
|
||||
} else {
|
||||
// All this must be done for all items manually in History::clear(false)!
|
||||
eraseFromUnreadMentions();
|
||||
|
@ -309,7 +311,7 @@ void HistoryItem::destroy() {
|
|||
const auto wasAtBottom = history->loadedAtBottom();
|
||||
history->removeNotification(this);
|
||||
|
||||
detach();
|
||||
removeMainView();
|
||||
|
||||
if (history->lastMsg == this) {
|
||||
history->fixLastMessage(wasAtBottom);
|
||||
|
@ -336,33 +338,18 @@ void HistoryItem::destroy() {
|
|||
delete this;
|
||||
}
|
||||
|
||||
void HistoryItem::detach() {
|
||||
if (detached()) return;
|
||||
|
||||
if (_history->isChannel()) {
|
||||
_history->asChannelHistory()->messageDetached(this);
|
||||
void HistoryItem::removeMainView() {
|
||||
if (const auto view = mainView()) {
|
||||
if (const auto channelHistory = _history->asChannelHistory()) {
|
||||
channelHistory->messageDetached(this);
|
||||
}
|
||||
_history->setPendingResize();
|
||||
view->removeFromBlock();
|
||||
_mainView = nullptr;
|
||||
}
|
||||
_block->removeItem(this);
|
||||
App::historyItemDetached(this);
|
||||
|
||||
_history->setPendingResize();
|
||||
}
|
||||
|
||||
void HistoryItem::attachToBlock(not_null<HistoryBlock*> block, int index) {
|
||||
Expects(!isLogEntry());
|
||||
Expects(_block == nullptr);
|
||||
Expects(_indexInBlock < 0);
|
||||
Expects(index >= 0);
|
||||
|
||||
_block = block;
|
||||
_indexInBlock = index;
|
||||
_mainView = block->messages[index].get();
|
||||
setPendingResize();
|
||||
}
|
||||
|
||||
void HistoryItem::detachFast() {
|
||||
_block = nullptr;
|
||||
_indexInBlock = -1;
|
||||
void HistoryItem::clearMainView() {
|
||||
_mainView = nullptr;
|
||||
|
||||
validateGroupId();
|
||||
|
@ -391,6 +378,7 @@ void HistoryItem::indexAsNewItem() {
|
|||
|
||||
void HistoryItem::previousItemChanged() {
|
||||
Expects(!isLogEntry());
|
||||
|
||||
recountDisplayDate();
|
||||
recountAttachToPrevious();
|
||||
}
|
||||
|
@ -398,6 +386,7 @@ void HistoryItem::previousItemChanged() {
|
|||
// Called only if there is no more next item! Not always when it changes!
|
||||
void HistoryItem::nextItemChanged() {
|
||||
Expects(!isLogEntry());
|
||||
|
||||
setAttachToNext(false);
|
||||
}
|
||||
|
||||
|
@ -421,6 +410,7 @@ bool HistoryItem::computeIsAttachToPrevious(not_null<HistoryItem*> previous) {
|
|||
|
||||
void HistoryItem::recountAttachToPrevious() {
|
||||
Expects(!isLogEntry());
|
||||
|
||||
auto attachToPrevious = false;
|
||||
if (auto previous = previousItem()) {
|
||||
attachToPrevious = computeIsAttachToPrevious(previous);
|
||||
|
@ -940,11 +930,6 @@ void HistoryItem::clipCallback(Media::Clip::Notification notification) {
|
|||
}
|
||||
if (!stopped) {
|
||||
setPendingInitDimensions();
|
||||
if (detached()) {
|
||||
// We still want to handle our pending initDimensions and
|
||||
// resize state even if we're detached in history.
|
||||
_history->setHasPendingResizedItems();
|
||||
}
|
||||
Auth().data().markItemLayoutChanged(this);
|
||||
Global::RefPendingRepaintItems().insert(this);
|
||||
}
|
||||
|
@ -984,27 +969,26 @@ void HistoryItem::audioTrackUpdated() {
|
|||
}
|
||||
}
|
||||
|
||||
bool HistoryItem::isUnderCursor() const {
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
return view->data() == this;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
HistoryItem *HistoryItem::previousItem() const {
|
||||
if (_block && _indexInBlock >= 0) {
|
||||
if (_indexInBlock > 0) {
|
||||
return _block->messages[_indexInBlock - 1]->data();
|
||||
}
|
||||
if (auto previous = _block->previousBlock()) {
|
||||
Assert(!previous->messages.empty());
|
||||
return previous->messages.back()->data();
|
||||
if (const auto view = mainView()) {
|
||||
if (const auto previous = view->previousInBlocks()) {
|
||||
return previous->data();
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
HistoryItem *HistoryItem::nextItem() const {
|
||||
if (_block && _indexInBlock >= 0) {
|
||||
if (_indexInBlock + 1 < _block->messages.size()) {
|
||||
return _block->messages[_indexInBlock + 1]->data();
|
||||
}
|
||||
if (auto next = _block->nextBlock()) {
|
||||
Assert(!next->messages.empty());
|
||||
return next->messages.front()->data();
|
||||
if (const auto view = mainView()) {
|
||||
if (const auto next = view->nextInBlocks()) {
|
||||
return next->data();
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
|
@ -1107,9 +1091,9 @@ HistoryItem::~HistoryItem() {
|
|||
ClickHandlerPtr goToMessageClickHandler(PeerData *peer, MsgId msgId) {
|
||||
return std::make_shared<LambdaClickHandler>([peer, msgId] {
|
||||
if (App::main()) {
|
||||
auto current = App::mousedItem();
|
||||
if (current && current->history()->peer == peer) {
|
||||
App::main()->pushReplyReturn(current);
|
||||
auto view = App::mousedItem();
|
||||
if (view && view->data()->history()->peer == peer) {
|
||||
App::main()->pushReplyReturn(view->data());
|
||||
}
|
||||
App::wnd()->controller()->showPeerHistory(
|
||||
peer,
|
||||
|
|
|
@ -53,6 +53,9 @@ public:
|
|||
int minHeight() const {
|
||||
return _minh;
|
||||
}
|
||||
int width() const {
|
||||
return _width;
|
||||
}
|
||||
int height() const {
|
||||
return _height;
|
||||
}
|
||||
|
@ -62,6 +65,7 @@ public:
|
|||
protected:
|
||||
mutable int _maxw = 0;
|
||||
mutable int _minh = 0;
|
||||
mutable int _width = 0;
|
||||
mutable int _height = 0;
|
||||
|
||||
};
|
||||
|
@ -226,34 +230,16 @@ public:
|
|||
PeerData *from() const {
|
||||
return _from;
|
||||
}
|
||||
HistoryBlock *block() {
|
||||
return _block;
|
||||
}
|
||||
const HistoryBlock *block() const {
|
||||
return _block;
|
||||
}
|
||||
HistoryView::Message *mainView() const {
|
||||
return _mainView;
|
||||
}
|
||||
void setMainView(HistoryView::Message *view) {
|
||||
_mainView = view;
|
||||
}
|
||||
void clearMainView();
|
||||
void removeMainView();
|
||||
|
||||
void destroy();
|
||||
void detach();
|
||||
void detachFast();
|
||||
bool detached() const {
|
||||
return !_block;
|
||||
}
|
||||
void attachToBlock(not_null<HistoryBlock*> block, int index);
|
||||
void setIndexInBlock(int index) {
|
||||
Expects(_block != nullptr);
|
||||
Expects(index >= 0);
|
||||
|
||||
_indexInBlock = index;
|
||||
}
|
||||
int indexInBlock() const {
|
||||
Expects((_indexInBlock >= 0) == (_block != nullptr));
|
||||
//Expects((_block == nullptr) || (_block->messages[_indexInBlock]->data() == this));
|
||||
|
||||
return _indexInBlock;
|
||||
}
|
||||
bool out() const {
|
||||
return _flags & MTPDmessage::Flag::f_out;
|
||||
}
|
||||
|
@ -493,9 +479,7 @@ public:
|
|||
}
|
||||
void setPendingResize() {
|
||||
_flags |= MTPDmessage_ClientFlag::f_pending_resize;
|
||||
if (!detached() || isLogEntry()) {
|
||||
_history->setHasPendingResizedItems();
|
||||
}
|
||||
_history->setHasPendingResizedItems();
|
||||
}
|
||||
bool pendingInitDimensions() const {
|
||||
return _flags & MTPDmessage_ClientFlag::f_pending_init_dimensions;
|
||||
|
@ -535,10 +519,6 @@ public:
|
|||
void makeGroupLeader(std::vector<not_null<HistoryItem*>> &&others);
|
||||
HistoryMessageGroup *getFullGroup();
|
||||
|
||||
int width() const {
|
||||
return _width;
|
||||
}
|
||||
|
||||
void clipCallback(Media::Clip::Notification notification);
|
||||
void audioTrackUpdated();
|
||||
|
||||
|
@ -556,6 +536,8 @@ public:
|
|||
setAttachToNext(attachToNext);
|
||||
}
|
||||
|
||||
bool isUnderCursor() const;
|
||||
|
||||
HistoryItem *previousItem() const;
|
||||
HistoryItem *nextItem() const;
|
||||
|
||||
|
@ -586,8 +568,6 @@ protected:
|
|||
|
||||
const not_null<History*> _history;
|
||||
not_null<PeerData*> _from;
|
||||
HistoryBlock *_block = nullptr;
|
||||
int _indexInBlock = -1;
|
||||
MTPDmessage::Flags _flags = 0;
|
||||
|
||||
// This should be called only from previousItemChanged()
|
||||
|
@ -642,7 +622,6 @@ private:
|
|||
void resetGroupMedia(const std::vector<not_null<HistoryItem*>> &others);
|
||||
|
||||
HistoryView::Message *_mainView = nullptr;
|
||||
int _width = 0;
|
||||
|
||||
};
|
||||
|
||||
|
|
|
@ -398,8 +398,9 @@ struct HistoryMessageLogEntryOriginal : public RuntimeComponent<HistoryMessageLo
|
|||
|
||||
};
|
||||
|
||||
class FileClickHandler;
|
||||
struct HistoryDocumentThumbed : public RuntimeComponent<HistoryDocumentThumbed> {
|
||||
ClickHandlerPtr _linksavel, _linkcancell;
|
||||
std::shared_ptr<FileClickHandler> _linksavel, _linkcancell;
|
||||
int _thumbw = 0;
|
||||
|
||||
mutable int _linkw = 0;
|
||||
|
|
|
@ -230,10 +230,6 @@ public:
|
|||
return QString();
|
||||
}
|
||||
|
||||
int currentWidth() const {
|
||||
return _width;
|
||||
}
|
||||
|
||||
void setInBubbleState(MediaInBubbleState state) {
|
||||
_inBubbleState = state;
|
||||
}
|
||||
|
@ -265,7 +261,6 @@ public:
|
|||
|
||||
protected:
|
||||
not_null<HistoryItem*> _parent;
|
||||
int _width = 0;
|
||||
MediaInBubbleState _inBubbleState = MediaInBubbleState::None;
|
||||
|
||||
|
||||
|
|
|
@ -10,6 +10,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|||
#include "history/history_item_components.h"
|
||||
#include "history/history_media_types.h"
|
||||
#include "history/history_message.h"
|
||||
#include "history/view/history_view_message.h"
|
||||
#include "storage/storage_shared_media.h"
|
||||
#include "lang/lang_keys.h"
|
||||
#include "ui/grouped_layout.h"
|
||||
|
@ -176,7 +177,7 @@ void HistoryGroupedMedia::draw(
|
|||
} else if (_parent->getMedia() == this) {
|
||||
auto fullRight = _width;
|
||||
auto fullBottom = _height;
|
||||
if (_parent->id < 0 || App::hoveredItem() == _parent) {
|
||||
if (needInfoDisplay()) {
|
||||
_parent->drawInfo(p, fullRight, fullBottom, _width, selected, InfoDisplayOverImage);
|
||||
}
|
||||
if (!_parent->hasBubble() && _parent->displayRightAction()) {
|
||||
|
@ -295,7 +296,8 @@ void HistoryGroupedMedia::clickHandlerPressedChanged(
|
|||
for (const auto &element : _elements) {
|
||||
element.content->clickHandlerPressedChanged(p, pressed);
|
||||
if (pressed && element.content->dragItemByHandler(p)) {
|
||||
App::pressedLinkItem(element.item);
|
||||
// #TODO pressedLinkItem
|
||||
//App::pressedLinkItem(element.item);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -455,3 +457,7 @@ bool HistoryGroupedMedia::computeNeedBubble() const {
|
|||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
bool HistoryGroupedMedia::needInfoDisplay() const {
|
||||
return (_parent->id < 0 || _parent->isUnderCursor());
|
||||
}
|
||||
|
|
|
@ -115,6 +115,7 @@ private:
|
|||
|
||||
};
|
||||
|
||||
bool needInfoDisplay() const;
|
||||
bool computeNeedBubble() const;
|
||||
not_null<HistoryMedia*> main() const;
|
||||
bool validateGroupElements(
|
||||
|
|
|
@ -446,7 +446,7 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim
|
|||
} else if (notChild) {
|
||||
auto fullRight = skipx + width;
|
||||
auto fullBottom = skipy + height;
|
||||
if (_data->uploading() || App::hoveredItem() == _parent) {
|
||||
if (needInfoDisplay()) {
|
||||
_parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, InfoDisplayOverImage);
|
||||
}
|
||||
if (!bubble && _parent->displayRightAction()) {
|
||||
|
@ -646,6 +646,10 @@ bool HistoryPhoto::dataLoaded() const {
|
|||
return _data->loaded();
|
||||
}
|
||||
|
||||
bool HistoryPhoto::needInfoDisplay() const {
|
||||
return (_data->uploading() || _parent->isUnderCursor());
|
||||
}
|
||||
|
||||
void HistoryPhoto::validateGroupedCache(
|
||||
const QRect &geometry,
|
||||
RectParts corners,
|
||||
|
@ -1402,11 +1406,17 @@ void HistoryDocument::createComponents(bool caption) {
|
|||
}
|
||||
UpdateComponents(mask);
|
||||
if (auto thumbed = Get<HistoryDocumentThumbed>()) {
|
||||
thumbed->_linksavel = std::make_shared<DocumentSaveClickHandler>(_data);
|
||||
thumbed->_linkcancell = std::make_shared<DocumentCancelClickHandler>(_data);
|
||||
thumbed->_linksavel = std::make_shared<DocumentSaveClickHandler>(
|
||||
_data,
|
||||
_parent->fullId());
|
||||
thumbed->_linkcancell = std::make_shared<DocumentCancelClickHandler>(
|
||||
_data,
|
||||
_parent->fullId());
|
||||
}
|
||||
if (auto voice = Get<HistoryDocumentVoice>()) {
|
||||
voice->_seekl = std::make_shared<VoiceSeekClickHandler>(_data);
|
||||
voice->_seekl = std::make_shared<VoiceSeekClickHandler>(
|
||||
_data,
|
||||
_parent->fullId());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -2066,6 +2076,23 @@ void HistoryDocument::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool
|
|||
HistoryFileMedia::clickHandlerPressedChanged(p, pressed);
|
||||
}
|
||||
|
||||
void HistoryDocument::refreshParentId(not_null<HistoryItem*> realParent) {
|
||||
HistoryFileMedia::refreshParentId(realParent);
|
||||
|
||||
const auto contextId = realParent->fullId();
|
||||
if (auto thumbed = Get<HistoryDocumentThumbed>()) {
|
||||
if (thumbed->_linksavel) {
|
||||
thumbed->_linksavel->setMessageId(contextId);
|
||||
thumbed->_linkcancell->setMessageId(contextId);
|
||||
}
|
||||
}
|
||||
if (auto voice = Get<HistoryDocumentVoice>()) {
|
||||
if (voice->_seekl) {
|
||||
voice->_seekl->setMessageId(contextId);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool HistoryDocument::playInline(bool autoplay) {
|
||||
if (_data->isVoiceMessage()) {
|
||||
DocumentOpenClickHandler::doOpen(_data, _parent, ActionOnLoadPlayInline);
|
||||
|
@ -2157,7 +2184,9 @@ void HistoryGif::initDimensions() {
|
|||
_caption.setSkipBlock(_parent->skipBlockWidth(), _parent->skipBlockHeight());
|
||||
}
|
||||
if (!_openInMediaviewLink) {
|
||||
_openInMediaviewLink = std::make_shared<DocumentOpenClickHandler>(_data);
|
||||
_openInMediaviewLink = std::make_shared<DocumentOpenClickHandler>(
|
||||
_data,
|
||||
_parent->fullId());
|
||||
}
|
||||
|
||||
int32 tw = 0, th = 0;
|
||||
|
@ -2548,7 +2577,7 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
|
|||
fullRight = maxRight;
|
||||
}
|
||||
}
|
||||
if (isRound || _data->uploading() || App::hoveredItem() == _parent) {
|
||||
if (isRound || needInfoDisplay()) {
|
||||
_parent->drawInfo(p, fullRight, fullBottom, 2 * skipx + width, selected, isRound ? InfoDisplayOverBackground : InfoDisplayOverImage);
|
||||
}
|
||||
if (!bubble && _parent->displayRightAction()) {
|
||||
|
@ -2955,6 +2984,10 @@ bool HistoryGif::dataLoaded() const {
|
|||
return (_parent->id > 0) ? _data->loaded() : false;
|
||||
}
|
||||
|
||||
bool HistoryGif::needInfoDisplay() const {
|
||||
return (_data->uploading() || _parent->isUnderCursor());
|
||||
}
|
||||
|
||||
HistorySticker::HistorySticker(
|
||||
not_null<HistoryItem*> parent,
|
||||
not_null<DocumentData*> document)
|
||||
|
@ -3835,7 +3868,7 @@ int HistoryWebPage::resizeGetHeight(int width) {
|
|||
_height += _attach->height() - bubble.top() - bubble.bottom();
|
||||
if (!_attach->additionalInfoString().isEmpty()) {
|
||||
_height += bottomInfoPadding();
|
||||
} else if (isBubbleBottom() && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
} else if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
_height += bottomInfoPadding();
|
||||
}
|
||||
}
|
||||
|
@ -3873,7 +3906,7 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T
|
|||
bshift += bottomInfoPadding();
|
||||
} else if (!attachAdditionalInfoText.isEmpty()) {
|
||||
bshift += bottomInfoPadding();
|
||||
} else if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
} else if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
|
||||
|
@ -3939,13 +3972,13 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T
|
|||
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
|
||||
p.translate(attachLeft, attachTop);
|
||||
|
||||
auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 };
|
||||
_attach->draw(p, r.translated(-attachLeft, -attachTop), attachSelection, ms);
|
||||
int32 pixwidth = _attach->currentWidth(), pixheight = _attach->height();
|
||||
int32 pixwidth = _attach->width(), pixheight = _attach->height();
|
||||
|
||||
if (_data->type == WebPageVideo && _attach->type() == MediaTypePhoto) {
|
||||
if (_attach->isReadyForOpen()) {
|
||||
|
@ -3991,7 +4024,7 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ
|
|||
auto padding = inBubblePadding();
|
||||
auto tshift = padding.top();
|
||||
auto bshift = padding.bottom();
|
||||
if (_asArticle || (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right())) {
|
||||
if (_asArticle || (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right())) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
width -= padding.left() + padding.right();
|
||||
|
@ -4055,7 +4088,7 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ
|
|||
if (QRect(padding.left(), tshift, width, _height - tshift - bshift).contains(point)) {
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
result = _attach->getState(point - QPoint(attachLeft, attachTop), request);
|
||||
|
||||
if (result.link && !_data->document && _data->photo && _attach->isReadyForOpen()) {
|
||||
|
@ -4325,7 +4358,7 @@ int HistoryGame::resizeGetHeight(int width) {
|
|||
|
||||
_attach->resizeGetHeight(width + bubble.left() + bubble.right());
|
||||
_height += _attach->height() - bubble.top() - bubble.bottom();
|
||||
if (isBubbleBottom() && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
_height += bottomInfoPadding();
|
||||
}
|
||||
}
|
||||
|
@ -4351,7 +4384,7 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time
|
|||
auto tshift = padding.top();
|
||||
auto bshift = padding.bottom();
|
||||
width -= padding.left() + padding.right();
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
|
||||
|
@ -4383,13 +4416,13 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time
|
|||
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
|
||||
auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 };
|
||||
|
||||
p.translate(attachLeft, attachTop);
|
||||
_attach->draw(p, r.translated(-attachLeft, -attachTop), attachSelection, ms);
|
||||
auto pixwidth = _attach->currentWidth();
|
||||
auto pixwidth = _attach->width();
|
||||
auto pixheight = _attach->height();
|
||||
|
||||
auto gameW = _gameTagWidth + 2 * st::msgDateImgPadding.x();
|
||||
|
@ -4419,7 +4452,7 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request
|
|||
auto padding = inBubblePadding();
|
||||
auto tshift = padding.top();
|
||||
auto bshift = padding.bottom();
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
width -= padding.left() + padding.right();
|
||||
|
@ -4465,9 +4498,9 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request
|
|||
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
|
||||
if (QRect(attachLeft, tshift, _attach->currentWidth(), _height - tshift - bshift).contains(point)) {
|
||||
if (QRect(attachLeft, tshift, _attach->width(), _height - tshift - bshift).contains(point)) {
|
||||
if (_attach->isReadyForOpen()) {
|
||||
if (!_parent->isLogEntry()) {
|
||||
result.link = _openl;
|
||||
|
@ -4782,7 +4815,7 @@ int HistoryInvoice::resizeGetHeight(int width) {
|
|||
|
||||
_attach->resizeGetHeight(width + bubble.left() + bubble.right());
|
||||
_height += _attach->height() - bubble.top() - bubble.bottom();
|
||||
if (isBubbleBottom() && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
_height += bottomInfoPadding();
|
||||
}
|
||||
} else {
|
||||
|
@ -4816,7 +4849,7 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T
|
|||
auto tshift = padding.top();
|
||||
auto bshift = padding.bottom();
|
||||
width -= padding.left() + padding.right();
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
|
||||
|
@ -4845,13 +4878,13 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T
|
|||
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
|
||||
auto attachSelection = selected ? FullSelection : TextSelection { 0, 0 };
|
||||
|
||||
p.translate(attachLeft, attachTop);
|
||||
_attach->draw(p, r.translated(-attachLeft, -attachTop), attachSelection, ms);
|
||||
auto pixwidth = _attach->currentWidth();
|
||||
auto pixwidth = _attach->width();
|
||||
auto pixheight = _attach->height();
|
||||
|
||||
auto available = _status.maxWidth();
|
||||
|
@ -4885,7 +4918,7 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ
|
|||
auto padding = inBubblePadding();
|
||||
auto tshift = padding.top();
|
||||
auto bshift = padding.bottom();
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->currentWidth() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
if (isBubbleBottom() && _attach && _attach->customInfoLayout() && _attach->width() + _parent->skipBlockWidth() > width + bubble.left() + bubble.right()) {
|
||||
bshift += bottomInfoPadding();
|
||||
}
|
||||
width -= padding.left() + padding.right();
|
||||
|
@ -4924,9 +4957,9 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ
|
|||
|
||||
auto attachLeft = padding.left() - bubble.left();
|
||||
auto attachTop = tshift - bubble.top();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->currentWidth();
|
||||
if (rtl()) attachLeft = _width - attachLeft - _attach->width();
|
||||
|
||||
if (QRect(attachLeft, tshift, _attach->currentWidth(), _height - tshift - bshift).contains(point)) {
|
||||
if (QRect(attachLeft, tshift, _attach->width(), _height - tshift - bshift).contains(point)) {
|
||||
result = _attach->getState(point - QPoint(attachLeft, attachTop), request);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -254,6 +254,7 @@ protected:
|
|||
bool dataLoaded() const override;
|
||||
|
||||
private:
|
||||
bool needInfoDisplay() const;
|
||||
void validateGroupedCache(
|
||||
const QRect &geometry,
|
||||
RectParts corners,
|
||||
|
@ -466,6 +467,8 @@ public:
|
|||
|
||||
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||
|
||||
void refreshParentId(not_null<HistoryItem*> realParent) override;
|
||||
|
||||
protected:
|
||||
float64 dataProgress() const override;
|
||||
bool dataFinished() const override;
|
||||
|
@ -588,6 +591,7 @@ protected:
|
|||
}
|
||||
|
||||
private:
|
||||
bool needInfoDisplay() const;
|
||||
int additionalWidth(
|
||||
const HistoryMessageVia *via,
|
||||
const HistoryMessageReply *reply,
|
||||
|
|
|
@ -1227,8 +1227,8 @@ bool HistoryMessage::displayFastReply() const {
|
|||
|
||||
QRect HistoryMessage::countGeometry() const {
|
||||
auto maxwidth = qMin(st::msgMaxWidth, _maxw);
|
||||
if (_media && _media->currentWidth() < maxwidth) {
|
||||
maxwidth = qMax(_media->currentWidth(), qMin(maxwidth, plainMaxWidth()));
|
||||
if (_media && _media->width() < maxwidth) {
|
||||
maxwidth = qMax(_media->width(), qMin(maxwidth, plainMaxWidth()));
|
||||
}
|
||||
|
||||
const auto outLayout = hasOutLayout();
|
||||
|
@ -1904,7 +1904,7 @@ void HistoryMessage::paintFromName(
|
|||
return 0;
|
||||
}();
|
||||
const auto replyWidth = [&] {
|
||||
if (App::hoveredItem() == this && displayFastReply()) {
|
||||
if (isUnderCursor() && displayFastReply()) {
|
||||
return st::msgFont->width(FastReplyText());
|
||||
}
|
||||
return 0;
|
||||
|
@ -2364,7 +2364,7 @@ bool HistoryMessage::getStateFromName(
|
|||
not_null<HistoryTextState*> outResult) const {
|
||||
if (displayFromName()) {
|
||||
const auto replyWidth = [&] {
|
||||
if (App::hoveredItem() == this && displayFastReply()) {
|
||||
if (isUnderCursor() && displayFastReply()) {
|
||||
return st::msgFont->width(FastReplyText());
|
||||
}
|
||||
return 0;
|
||||
|
|
|
@ -550,7 +550,7 @@ int HistoryService::resizeContentGetHeight() {
|
|||
}
|
||||
_height += st::msgServicePadding.top() + st::msgServicePadding.bottom() + st::msgServiceMargin.top() + st::msgServiceMargin.bottom();
|
||||
if (_media) {
|
||||
_height += st::msgServiceMargin.top() + _media->resizeGetHeight(_media->currentWidth());
|
||||
_height += st::msgServiceMargin.top() + _media->resizeGetHeight(_media->width());
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -628,7 +628,7 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null<Window::Controller*> cont
|
|||
}));
|
||||
subscribe(Auth().data().pendingHistoryResize(), [this] { handlePendingHistoryUpdate(); });
|
||||
subscribe(Auth().data().queryItemVisibility(), [this](const Data::Session::ItemVisibilityQuery &query) {
|
||||
if (_a_show.animating() || _history != query.item->history() || query.item->detached() || !isVisible()) {
|
||||
if (_a_show.animating() || _history != query.item->history() || !query.item->mainView() || !isVisible()) {
|
||||
return;
|
||||
}
|
||||
auto top = _list->itemTop(query.item);
|
||||
|
@ -642,9 +642,7 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null<Window::Controller*> cont
|
|||
Auth().data().itemLayoutChanged(
|
||||
) | rpl::start_with_next([this](auto item) {
|
||||
if (_peer && _list) {
|
||||
if ((item == App::mousedItem())
|
||||
|| (item == App::hoveredItem())
|
||||
|| (item == App::hoveredLinkItem())) {
|
||||
if (item->isUnderCursor()) {
|
||||
_list->onUpdateSelected();
|
||||
}
|
||||
}
|
||||
|
@ -804,7 +802,7 @@ void HistoryWidget::adjustHighlightedMessageToMigrated() {
|
|||
_history->channelId(),
|
||||
_highlightedMessageId);
|
||||
if (highlighted && highlighted->isGroupMigrate()) {
|
||||
_highlightedMessageId = -_migrated->blocks.back()->messages.back()->id();
|
||||
_highlightedMessageId = -_migrated->blocks.back()->messages.back()->data()->id;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -818,7 +816,7 @@ void HistoryWidget::checkNextHighlight() {
|
|||
auto msgId = _highlightQueue.front();
|
||||
_highlightQueue.pop_front();
|
||||
auto item = getItemFromHistoryOrMigrated(msgId);
|
||||
if (item && !item->detached()) {
|
||||
if (item && item->mainView()) {
|
||||
return msgId;
|
||||
}
|
||||
}
|
||||
|
@ -832,7 +830,7 @@ void HistoryWidget::checkNextHighlight() {
|
|||
|
||||
void HistoryWidget::updateHighlightedMessage() {
|
||||
auto item = getItemFromHistoryOrMigrated(_highlightedMessageId);
|
||||
if (!item || item->detached()) {
|
||||
if (!item || !item->mainView()) {
|
||||
return stopMessageHighlight();
|
||||
}
|
||||
auto duration = st::activeFadeInDuration + st::activeFadeOutDuration;
|
||||
|
@ -1440,8 +1438,7 @@ void HistoryWidget::clearReplyReturns() {
|
|||
_replyReturn = 0;
|
||||
}
|
||||
|
||||
void HistoryWidget::pushReplyReturn(HistoryItem *item) {
|
||||
if (!item) return;
|
||||
void HistoryWidget::pushReplyReturn(not_null<HistoryItem*> item) {
|
||||
if (item->history() == _history) {
|
||||
_replyReturns.push_back(item->id);
|
||||
} else if (item->history() == _migrated) {
|
||||
|
@ -1707,13 +1704,6 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
|
|||
_nonEmptySelection = false;
|
||||
_topBar->showSelected(HistoryView::TopBarWidget::SelectedState {});
|
||||
|
||||
App::hoveredItem(nullptr);
|
||||
App::pressedItem(nullptr);
|
||||
App::hoveredLinkItem(nullptr);
|
||||
App::pressedLinkItem(nullptr);
|
||||
App::contextItem(nullptr);
|
||||
App::mousedItem(nullptr);
|
||||
|
||||
if (_peer) {
|
||||
App::forgetMedia();
|
||||
_serviceImageCacheSize = imageCacheSize();
|
||||
|
@ -2424,7 +2414,7 @@ bool HistoryWidget::doWeReadServerHistory() const {
|
|||
if (scrollTop + 1 > _scroll->scrollTopMax()) return true;
|
||||
|
||||
auto showFrom = (_migrated && _migrated->showFrom) ? _migrated->showFrom : (_history ? _history->showFrom : nullptr);
|
||||
if (showFrom && !showFrom->detached()) {
|
||||
if (showFrom && showFrom->mainView()) {
|
||||
int scrollBottom = scrollTop + _scroll->height();
|
||||
if (scrollBottom > _list->itemTop(showFrom)) return true;
|
||||
}
|
||||
|
@ -2445,7 +2435,7 @@ bool HistoryWidget::doWeReadMentions() const {
|
|||
}
|
||||
|
||||
bool HistoryWidget::historyHasNotFreezedUnreadBar(History *history) const {
|
||||
if (history && history->showFrom && !history->showFrom->detached() && history->unreadBar) {
|
||||
if (history && history->showFrom && history->showFrom->mainView() && history->unreadBar) {
|
||||
if (auto unreadBar = history->unreadBar->Get<HistoryMessageUnreadBar>()) {
|
||||
return !unreadBar->_freezed;
|
||||
}
|
||||
|
@ -2675,7 +2665,7 @@ void HistoryWidget::visibleAreaUpdated() {
|
|||
_list->visibleAreaUpdated(scrollTop, scrollBottom);
|
||||
if (_history->loadedAtBottom() && (_history->unreadCount() > 0 || (_migrated && _migrated->unreadCount() > 0))) {
|
||||
auto showFrom = (_migrated && _migrated->showFrom) ? _migrated->showFrom : (_history ? _history->showFrom : nullptr);
|
||||
auto showFromVisible = (showFrom && !showFrom->detached() && scrollBottom > _list->itemTop(showFrom));
|
||||
auto showFromVisible = (showFrom && showFrom->mainView() && scrollBottom > _list->itemTop(showFrom));
|
||||
auto atBottom = (scrollTop >= _scroll->scrollTopMax());
|
||||
if ((showFromVisible || atBottom) && App::wnd()->doWeReadServerHistory()) {
|
||||
Auth().api().readServerHistory(_history);
|
||||
|
@ -2727,14 +2717,14 @@ void HistoryWidget::checkReplyReturns() {
|
|||
auto scrollTopMax = _scroll->scrollTopMax();
|
||||
auto scrollHeight = _scroll->height();
|
||||
while (_replyReturn) {
|
||||
auto below = (_replyReturn->detached() && _replyReturn->history() == _history && !_history->isEmpty() && _replyReturn->id < _history->blocks.back()->messages.back()->id());
|
||||
auto below = (!_replyReturn->mainView() && _replyReturn->history() == _history && !_history->isEmpty() && _replyReturn->id < _history->blocks.back()->messages.back()->data()->id);
|
||||
if (!below) {
|
||||
below = (_replyReturn->detached() && _replyReturn->history() == _migrated && !_history->isEmpty());
|
||||
below = (!_replyReturn->mainView() && _replyReturn->history() == _migrated && !_history->isEmpty());
|
||||
}
|
||||
if (!below) {
|
||||
below = (_replyReturn->detached() && _migrated && _replyReturn->history() == _migrated && !_migrated->isEmpty() && _replyReturn->id < _migrated->blocks.back()->messages.back()->id());
|
||||
below = (!_replyReturn->mainView() && _migrated && _replyReturn->history() == _migrated && !_migrated->isEmpty() && _replyReturn->id < _migrated->blocks.back()->messages.back()->data()->id);
|
||||
}
|
||||
if (!below && !_replyReturn->detached()) {
|
||||
if (!below && _replyReturn->mainView()) {
|
||||
below = (scrollTop >= scrollTopMax) || (_list->itemTop(_replyReturn) < scrollTop + scrollHeight / 2);
|
||||
}
|
||||
if (below) {
|
||||
|
@ -3424,7 +3414,7 @@ bool HistoryWidget::insertBotCommand(const QString &cmd) {
|
|||
auto insertingInlineBot = !cmd.isEmpty() && (cmd.at(0) == '@');
|
||||
auto toInsert = cmd;
|
||||
if (!toInsert.isEmpty() && !insertingInlineBot) {
|
||||
auto bot = _peer->isUser() ? _peer : (App::hoveredLinkItem() ? App::hoveredLinkItem()->fromOriginal() : nullptr);
|
||||
auto bot = _peer->isUser() ? _peer : (App::hoveredLinkItem() ? App::hoveredLinkItem()->data()->fromOriginal() : nullptr);
|
||||
if (bot && (!bot->isUser() || !bot->asUser()->botInfo)) {
|
||||
bot = nullptr;
|
||||
}
|
||||
|
@ -4951,7 +4941,7 @@ void HistoryWidget::updateHistoryDownVisibility() {
|
|||
if (!_list || !history || history->unreadCount() <= 0) {
|
||||
return false;
|
||||
}
|
||||
if (!history->showFrom || history->showFrom->detached()) {
|
||||
if (!history->showFrom || !history->showFrom->mainView()) {
|
||||
return false;
|
||||
}
|
||||
return (_list->itemTop(history->showFrom) >= _scroll->scrollTop() + _scroll->height());
|
||||
|
@ -5052,41 +5042,34 @@ void HistoryWidget::keyPressEvent(QKeyEvent *e) {
|
|||
_scroll->keyPressEvent(e);
|
||||
} else if ((e->modifiers() & (Qt::ShiftModifier | Qt::MetaModifier | Qt::ControlModifier)) == Qt::ControlModifier) {
|
||||
if (_history && _history->lastMsg && !_editMsgId) {
|
||||
if (_replyToId) {
|
||||
HistoryItem *item = App::histItemById(_history->channelId(), _replyToId)->nextItem();
|
||||
if (item) App::contextItem(item);
|
||||
else { cancelReply(); return; }
|
||||
} else {
|
||||
return;
|
||||
if (const auto item = App::histItemById(_history->channelId(), _replyToId)) {
|
||||
if (const auto next = item->nextItem()) {
|
||||
Ui::showPeerHistory(_peer, next->id);
|
||||
replyToMessage(next);
|
||||
}
|
||||
}
|
||||
Ui::showPeerHistory(_peer, App::contextItem()->id);
|
||||
onReplyToMessage();
|
||||
return;
|
||||
}
|
||||
}
|
||||
} else if (e->key() == Qt::Key_Up) {
|
||||
if (!(e->modifiers() & (Qt::ShiftModifier | Qt::MetaModifier | Qt::ControlModifier))) {
|
||||
if (_history && _history->lastSentMsg && _history->lastSentMsg->canEdit(::date(unixtime()))) {
|
||||
if (_field->isEmpty() && !_editMsgId && !_replyToId) {
|
||||
App::contextItem(_history->lastSentMsg);
|
||||
onEditMessage();
|
||||
if (_field->isEmpty() && !_editMsgId && !_replyToId && _history->lastSentMsg) {
|
||||
editMessage(_history->lastSentMsg);
|
||||
return;
|
||||
}
|
||||
}
|
||||
_scroll->keyPressEvent(e);
|
||||
} else if ((e->modifiers() & (Qt::ShiftModifier | Qt::MetaModifier | Qt::ControlModifier)) == Qt::ControlModifier) {
|
||||
if (_history && _history->lastMsg && !_editMsgId) {
|
||||
if (_replyToId) {
|
||||
HistoryItem *item = App::histItemById(_history->channelId(), _replyToId);
|
||||
App::contextItem(item->previousItem());
|
||||
} else {
|
||||
App::contextItem(_history->lastMsg);
|
||||
if (const auto item = App::histItemById(_history->channelId(), _replyToId)) {
|
||||
if (const auto previous = item->previousItem()) {
|
||||
Ui::showPeerHistory(_peer, previous->id);
|
||||
replyToMessage(previous);
|
||||
}
|
||||
} else if (const auto previous = _history->lastMsg) {
|
||||
Ui::showPeerHistory(_peer, previous->id);
|
||||
replyToMessage(previous);
|
||||
}
|
||||
if (App::contextItem()) {
|
||||
Ui::showPeerHistory(_peer, App::contextItem()->id);
|
||||
onReplyToMessage();
|
||||
}
|
||||
return;
|
||||
}
|
||||
}
|
||||
} else if (e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter) {
|
||||
|
@ -5517,26 +5500,31 @@ void HistoryWidget::setFieldText(const TextWithTags &textWithTags, TextUpdateEve
|
|||
_previewLinks.clear();
|
||||
}
|
||||
|
||||
void HistoryWidget::onReplyToMessage() {
|
||||
auto to = App::contextItem();
|
||||
if (!to || to->id <= 0 || !_canSendMessages) return;
|
||||
void HistoryWidget::replyToMessage(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
replyToMessage(item);
|
||||
}
|
||||
}
|
||||
|
||||
if (to->history() == _migrated) {
|
||||
if (to->isGroupMigrate() && !_history->isEmpty() && _history->blocks.front()->messages.front()->data()->isGroupMigrate() && _history != _migrated) {
|
||||
App::contextItem(_history->blocks.front()->messages.front()->data());
|
||||
onReplyToMessage();
|
||||
App::contextItem(to);
|
||||
void HistoryWidget::replyToMessage(not_null<HistoryItem*> item) {
|
||||
if (!IsServerMsgId(item->id) || !_canSendMessages) {
|
||||
return;
|
||||
}
|
||||
if (item->history() == _migrated) {
|
||||
if (item->isGroupMigrate()
|
||||
&& !_history->isEmpty()
|
||||
&& _history->blocks.front()->messages.front()->data()->isGroupMigrate()
|
||||
&& _history != _migrated) {
|
||||
replyToMessage(_history->blocks.front()->messages.front()->data());
|
||||
} else {
|
||||
if (to->id < 0 || to->serviceMsg()) {
|
||||
if (item->serviceMsg()) {
|
||||
Ui::show(Box<InformBox>(lang(lng_reply_cant)));
|
||||
} else {
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_reply_cant_forward), lang(lng_selected_forward), base::lambda_guarded(this, [this] {
|
||||
auto item = App::contextItem();
|
||||
if (!item || item->id < 0 || item->serviceMsg()) return;
|
||||
|
||||
const auto itemId = item->fullId();
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_reply_cant_forward), lang(lng_selected_forward), base::lambda_guarded(this, [=] {
|
||||
App::main()->setForwardDraft(
|
||||
_peer->id,
|
||||
{ 1, item->fullId() });
|
||||
{ 1, itemId });
|
||||
})));
|
||||
}
|
||||
}
|
||||
|
@ -5547,13 +5535,17 @@ void HistoryWidget::onReplyToMessage() {
|
|||
|
||||
if (_editMsgId) {
|
||||
if (auto localDraft = _history->localDraft()) {
|
||||
localDraft->msgId = to->id;
|
||||
localDraft->msgId = item->id;
|
||||
} else {
|
||||
_history->setLocalDraft(std::make_unique<Data::Draft>(TextWithTags(), to->id, MessageCursor(), false));
|
||||
_history->setLocalDraft(std::make_unique<Data::Draft>(
|
||||
TextWithTags(),
|
||||
item->id,
|
||||
MessageCursor(),
|
||||
false));
|
||||
}
|
||||
} else {
|
||||
_replyEditMsg = to;
|
||||
_replyToId = to->id;
|
||||
_replyEditMsg = item;
|
||||
_replyToId = item->id;
|
||||
_replyEditMsgText.setText(
|
||||
st::messageTextStyle,
|
||||
TextUtilities::Clean(_replyEditMsg->inReplyText()),
|
||||
|
@ -5575,13 +5567,16 @@ void HistoryWidget::onReplyToMessage() {
|
|||
_field->setFocus();
|
||||
}
|
||||
|
||||
void HistoryWidget::onEditMessage() {
|
||||
auto to = App::contextItem();
|
||||
if (!to) return;
|
||||
void HistoryWidget::editMessage(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
editMessage(item);
|
||||
}
|
||||
}
|
||||
|
||||
if (auto media = to->getMedia()) {
|
||||
void HistoryWidget::editMessage(not_null<HistoryItem*> item) {
|
||||
if (const auto media = item->getMedia()) {
|
||||
if (media->canEditCaption()) {
|
||||
Ui::show(Box<EditCaptionBox>(media, to->fullId()));
|
||||
Ui::show(Box<EditCaptionBox>(media, item->fullId()));
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
@ -5598,14 +5593,25 @@ void HistoryWidget::onEditMessage() {
|
|||
}
|
||||
}
|
||||
|
||||
auto original = to->originalText();
|
||||
auto editData = TextWithTags { TextUtilities::ApplyEntities(original), ConvertEntitiesToTextTags(original.entities) };
|
||||
auto cursor = MessageCursor { editData.text.size(), editData.text.size(), QFIXED_MAX };
|
||||
_history->setEditDraft(std::make_unique<Data::Draft>(editData, to->id, cursor, false));
|
||||
const auto original = item->originalText();
|
||||
const auto editData = TextWithTags {
|
||||
TextUtilities::ApplyEntities(original),
|
||||
ConvertEntitiesToTextTags(original.entities)
|
||||
};
|
||||
const auto cursor = MessageCursor {
|
||||
editData.text.size(),
|
||||
editData.text.size(),
|
||||
QFIXED_MAX
|
||||
};
|
||||
_history->setEditDraft(std::make_unique<Data::Draft>(
|
||||
editData,
|
||||
item->id,
|
||||
cursor,
|
||||
false));
|
||||
applyDraft(false);
|
||||
|
||||
_previewData = nullptr;
|
||||
if (auto media = to->getMedia()) {
|
||||
if (const auto media = item->getMedia()) {
|
||||
if (media->type() == MediaTypeWebPage) {
|
||||
_previewData = static_cast<HistoryWebPage*>(media)->webpage();
|
||||
updatePreview();
|
||||
|
@ -5631,22 +5637,23 @@ void HistoryWidget::onEditMessage() {
|
|||
_field->setFocus();
|
||||
}
|
||||
|
||||
void HistoryWidget::onPinMessage() {
|
||||
auto to = App::contextItem();
|
||||
if (!to || !to->canPin()) return;
|
||||
|
||||
Ui::show(Box<PinMessageBox>(
|
||||
to->history()->peer->asChannel(),
|
||||
to->id));
|
||||
void HistoryWidget::pinMessage(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (item->canPin()) {
|
||||
const auto channel = item->history()->peer->asChannel();
|
||||
Assert(channel != nullptr);
|
||||
Ui::show(Box<PinMessageBox>(channel, item->id));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void HistoryWidget::onUnpinMessage() {
|
||||
if (!_peer || !_peer->isChannel()) return;
|
||||
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_pinned_unpin_sure), lang(lng_pinned_unpin), base::lambda_guarded(this, [this] {
|
||||
auto channel = _peer ? _peer->asChannel() : nullptr;
|
||||
if (!channel) return;
|
||||
void HistoryWidget::unpinMessage(FullMsgId itemId) {
|
||||
const auto channel = _peer ? _peer->asChannel() : nullptr;
|
||||
if (!channel) {
|
||||
return;
|
||||
}
|
||||
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_pinned_unpin_sure), lang(lng_pinned_unpin), base::lambda_guarded(this, [=] {
|
||||
channel->clearPinnedMessage();
|
||||
|
||||
Ui::hideLayer();
|
||||
|
@ -5666,8 +5673,8 @@ void HistoryWidget::unpinDone(const MTPUpdates &updates) {
|
|||
}
|
||||
|
||||
void HistoryWidget::onPinnedHide() {
|
||||
auto channel = _peer ? _peer->asChannel() : nullptr;
|
||||
auto pinnedId = channel->pinnedMessageId();
|
||||
const auto channel = _peer ? _peer->asChannel() : nullptr;
|
||||
const auto pinnedId = channel ? channel->pinnedMessageId() : MsgId(0);
|
||||
if (!pinnedId) {
|
||||
if (pinnedMsgVisibilityUpdated()) {
|
||||
updateControlsGeometry();
|
||||
|
@ -5677,9 +5684,9 @@ void HistoryWidget::onPinnedHide() {
|
|||
}
|
||||
|
||||
if (channel->canPinMessages()) {
|
||||
onUnpinMessage();
|
||||
unpinMessage(FullMsgId(peerToChannel(channel->id), pinnedId));
|
||||
} else {
|
||||
Global::RefHiddenPinnedMessages().insert(_peer->id, pinnedId);
|
||||
Global::RefHiddenPinnedMessages().insert(channel->id, pinnedId);
|
||||
Local::writeUserSettings();
|
||||
if (pinnedMsgVisibilityUpdated()) {
|
||||
updateControlsGeometry();
|
||||
|
@ -5688,11 +5695,12 @@ void HistoryWidget::onPinnedHide() {
|
|||
}
|
||||
}
|
||||
|
||||
void HistoryWidget::onCopyPostLink() {
|
||||
auto item = App::contextItem();
|
||||
if (!item || !item->hasDirectLink()) return;
|
||||
|
||||
QApplication::clipboard()->setText(item->directLink());
|
||||
void HistoryWidget::copyPostLink(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (item->hasDirectLink()) {
|
||||
QApplication::clipboard()->setText(item->directLink());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool HistoryWidget::lastForceReplyReplied(const FullMsgId &replyTo) const {
|
||||
|
@ -6076,10 +6084,7 @@ void HistoryWidget::onForwardSelected() {
|
|||
void HistoryWidget::confirmDeleteSelectedItems() {
|
||||
if (!_list) return;
|
||||
|
||||
auto selected = _list->getSelectedItems();
|
||||
if (selected.empty()) return;
|
||||
|
||||
App::main()->deleteLayer(int(selected.size()));
|
||||
App::main()->deleteLayer(_list->getSelectedItems());
|
||||
}
|
||||
|
||||
void HistoryWidget::deleteSelectedItems(bool forEveryone) {
|
||||
|
|
|
@ -250,6 +250,14 @@ public:
|
|||
|
||||
void updateScrollColors();
|
||||
|
||||
void replyToMessage(FullMsgId itemId);
|
||||
void replyToMessage(not_null<HistoryItem*> item);
|
||||
void editMessage(FullMsgId itemId);
|
||||
void editMessage(not_null<HistoryItem*> item);
|
||||
void pinMessage(FullMsgId itemId);
|
||||
void unpinMessage(FullMsgId itemId);
|
||||
void copyPostLink(FullMsgId itemId);
|
||||
|
||||
MsgId replyToId() const;
|
||||
void messageDataReceived(ChannelData *channel, MsgId msgId);
|
||||
bool lastForceReplyReplied(const FullMsgId &replyTo) const;
|
||||
|
@ -260,7 +268,7 @@ public:
|
|||
void updateForwardingTexts();
|
||||
|
||||
void clearReplyReturns();
|
||||
void pushReplyReturn(HistoryItem *item);
|
||||
void pushReplyReturn(not_null<HistoryItem*> item);
|
||||
QList<MsgId> replyReturns();
|
||||
void setReplyReturns(PeerId peer, const QList<MsgId> &replyReturns);
|
||||
void calcNextReplyReturn();
|
||||
|
@ -361,12 +369,7 @@ signals:
|
|||
|
||||
public slots:
|
||||
void onCancel();
|
||||
void onReplyToMessage();
|
||||
void onEditMessage();
|
||||
void onPinMessage();
|
||||
void onUnpinMessage();
|
||||
void onPinnedHide();
|
||||
void onCopyPostLink();
|
||||
void onFieldBarCancel();
|
||||
|
||||
void onPreviewParse();
|
||||
|
|
|
@ -466,13 +466,13 @@ void ListWidget::checkMoveToOtherViewer() {
|
|||
|
||||
QString ListWidget::tooltipText() const {
|
||||
if (_mouseCursorState == HistoryInDateCursorState && _mouseAction == MouseAction::None) {
|
||||
if (const auto item = App::hoveredItem()) {
|
||||
auto dateText = item->date.toString(QLocale::system().dateTimeFormat(QLocale::LongFormat));
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
auto dateText = view->data()->date.toString(QLocale::system().dateTimeFormat(QLocale::LongFormat));
|
||||
return dateText;
|
||||
}
|
||||
} else if (_mouseCursorState == HistoryInForwardedCursorState && _mouseAction == MouseAction::None) {
|
||||
if (const auto item = App::hoveredItem()) {
|
||||
if (const auto forwarded = item->Get<HistoryMessageForwarded>()) {
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
if (const auto forwarded = view->data()->Get<HistoryMessageForwarded>()) {
|
||||
return forwarded->text.originalText(AllTextSelection, ExpandLinksNone);
|
||||
}
|
||||
}
|
||||
|
@ -724,10 +724,6 @@ void ListWidget::contextMenuEvent(QContextMenuEvent *e) {
|
|||
}
|
||||
|
||||
void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
||||
if (_menu) {
|
||||
_menu->deleteLater();
|
||||
_menu = 0;
|
||||
}
|
||||
if (e->reason() == QContextMenuEvent::Mouse) {
|
||||
mouseActionUpdate(e->globalPos());
|
||||
}
|
||||
|
@ -742,10 +738,12 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
auto selTo = _selectedText.to;
|
||||
hasSelected = (selTo > selFrom) ? 1 : 0;
|
||||
if (App::mousedItem() && App::mousedItem() == App::hoveredItem()) {
|
||||
auto mousePos = mapPointToItem(mapFromGlobal(_mousePosition), viewForItem(App::mousedItem()));
|
||||
auto mousePos = mapPointToItem(
|
||||
mapFromGlobal(_mousePosition),
|
||||
App::mousedItem());
|
||||
HistoryStateRequest request;
|
||||
request.flags |= Text::StateRequest::Flag::LookupSymbol;
|
||||
auto dragState = App::mousedItem()->getState(mousePos, request);
|
||||
auto dragState = App::mousedItem()->data()->getState(mousePos, request);
|
||||
if (dragState.cursor == HistoryInTextCursorState && dragState.symbol >= selFrom && dragState.symbol < selTo) {
|
||||
isUponSelected = 1;
|
||||
}
|
||||
|
@ -755,10 +753,12 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
isUponSelected = hasSelected;
|
||||
}
|
||||
|
||||
_menu = new Ui::PopupMenu(nullptr);
|
||||
_menu = base::make_unique_q<Ui::PopupMenu>(nullptr);
|
||||
|
||||
_contextMenuLink = ClickHandler::getActive();
|
||||
auto item = App::hoveredItem() ? App::hoveredItem() : App::hoveredLinkItem();
|
||||
auto view = App::hoveredItem()
|
||||
? App::hoveredItem()
|
||||
: App::hoveredLinkItem();
|
||||
auto lnkPhoto = dynamic_cast<PhotoClickHandler*>(_contextMenuLink.get());
|
||||
auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get());
|
||||
auto lnkPeer = dynamic_cast<PeerClickHandler*>(_contextMenuLink.get());
|
||||
|
@ -767,39 +767,50 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
auto lnkIsAudio = lnkDocument ? lnkDocument->document()->isAudioFile() : false;
|
||||
if (lnkPhoto || lnkDocument) {
|
||||
if (isUponSelected > 0) {
|
||||
_menu->addAction(lang(lng_context_copy_selected), [this] { copySelectedText(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_selected), [=] {
|
||||
copySelectedText();
|
||||
})->setEnabled(true);
|
||||
}
|
||||
if (lnkPhoto) {
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, photo = lnkPhoto->photo()] {
|
||||
const auto photo = lnkPhoto->photo();
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [=] {
|
||||
savePhotoToFile(photo);
|
||||
}))->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_image), [this, photo = lnkPhoto->photo()] {
|
||||
_menu->addAction(lang(lng_context_copy_image), [=] {
|
||||
copyContextImage(photo);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
auto document = lnkDocument->document();
|
||||
if (document->loading()) {
|
||||
_menu->addAction(lang(lng_context_cancel_download), [this] { cancelContextDownload(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_cancel_download), [=] {
|
||||
cancelContextDownload(document);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
if (document->loaded() && document->isGifv()) {
|
||||
if (!cAutoPlayGif()) {
|
||||
_menu->addAction(lang(lng_context_open_gif), [this] { openContextGif(); })->setEnabled(true);
|
||||
const auto itemId = view
|
||||
? view->data()->fullId()
|
||||
: FullMsgId();
|
||||
_menu->addAction(lang(lng_context_open_gif), [=] {
|
||||
openContextGif(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (!document->filepath(DocumentData::FilePathResolveChecked).isEmpty()) {
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [this] { showContextInFolder(); })->setEnabled(true);
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [=] {
|
||||
showContextInFolder(document);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
_menu->addAction(lang(lnkIsVideo ? lng_context_save_video : (lnkIsVoice ? lng_context_save_audio : (lnkIsAudio ? lng_context_save_audio_file : lng_context_save_file))), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
}))->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (App::hoveredLinkItem()) {
|
||||
App::contextItem(App::hoveredLinkItem());
|
||||
}
|
||||
} else if (lnkPeer) { // suggest to block
|
||||
// #TODO suggest restrict peer
|
||||
} else { // maybe cursor on some text history item?
|
||||
const auto item = view ? view->data().get() : nullptr;
|
||||
const auto itemId = item ? item->fullId() : FullMsgId();
|
||||
bool canDelete = item && item->canDelete() && (item->id > 0 || !item->serviceMsg());
|
||||
bool canForward = item && item->canForward();
|
||||
|
||||
|
@ -817,7 +828,9 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
if (media->type() == MediaTypeSticker) {
|
||||
if (auto document = media->getDocument()) {
|
||||
if (document->sticker() && document->sticker()->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
_menu->addAction(lang(document->sticker()->setInstalled() ? lng_context_pack_info : lng_context_pack_add), [this] { showStickerPackInfo(); });
|
||||
_menu->addAction(lang(document->sticker()->setInstalled() ? lng_context_pack_info : lng_context_pack_add), [=] {
|
||||
showStickerPackInfo(document);
|
||||
});
|
||||
}
|
||||
_menu->addAction(lang(lng_context_save_image), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
|
@ -826,15 +839,21 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
} else if (media->type() == MediaTypeGif && !_contextMenuLink) {
|
||||
if (auto document = media->getDocument()) {
|
||||
if (document->loading()) {
|
||||
_menu->addAction(lang(lng_context_cancel_download), [this] { cancelContextDownload(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_cancel_download), [=] {
|
||||
cancelContextDownload(document);
|
||||
})->setEnabled(true);
|
||||
} else {
|
||||
if (document->isGifv()) {
|
||||
if (!cAutoPlayGif()) {
|
||||
_menu->addAction(lang(lng_context_open_gif), [this] { openContextGif(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_open_gif), [=] {
|
||||
openContextGif(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
if (!document->filepath(DocumentData::FilePathResolveChecked).isEmpty()) {
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [this] { showContextInFolder(); })->setEnabled(true);
|
||||
_menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), [=] {
|
||||
showContextInFolder(document);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
_menu->addAction(lang(lng_context_save_file), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] {
|
||||
saveDocumentToFile(document);
|
||||
|
@ -844,7 +863,9 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
}
|
||||
}
|
||||
if (msg && !_contextMenuLink && (!msg->emptyText() || mediaHasTextForCopy)) {
|
||||
_menu->addAction(lang(lng_context_copy_text), [this] { copyContextText(); })->setEnabled(true);
|
||||
_menu->addAction(lang(lng_context_copy_text), [=] {
|
||||
copyContextText(itemId);
|
||||
})->setEnabled(true);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -853,17 +874,11 @@ void ListWidget::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
|
|||
if (!linkCopyToClipboardText.isEmpty()) {
|
||||
_menu->addAction(linkCopyToClipboardText, [this] { copyContextUrl(); })->setEnabled(true);
|
||||
}
|
||||
App::contextItem(item);
|
||||
}
|
||||
|
||||
if (_menu->actions().isEmpty()) {
|
||||
delete base::take(_menu);
|
||||
_menu = nullptr;
|
||||
} else {
|
||||
connect(_menu, &QObject::destroyed, this, [this](QObject *object) {
|
||||
if (_menu == object) {
|
||||
_menu = nullptr;
|
||||
}
|
||||
});
|
||||
_menu->popup(e->globalPos());
|
||||
e->accept();
|
||||
}
|
||||
|
@ -873,11 +888,15 @@ void ListWidget::savePhotoToFile(PhotoData *photo) {
|
|||
if (!photo || !photo->date || !photo->loaded()) return;
|
||||
|
||||
auto filter = qsl("JPEG Image (*.jpg);;") + FileDialog::AllFilesFilter();
|
||||
FileDialog::GetWritePath(lang(lng_save_photo), filter, filedialogDefaultName(qsl("photo"), qsl(".jpg")), base::lambda_guarded(this, [this, photo](const QString &result) {
|
||||
if (!result.isEmpty()) {
|
||||
photo->full->pix().toImage().save(result, "JPG");
|
||||
}
|
||||
}));
|
||||
FileDialog::GetWritePath(
|
||||
lang(lng_save_photo),
|
||||
filter,
|
||||
filedialogDefaultName(qsl("photo"), qsl(".jpg")),
|
||||
base::lambda_guarded(this, [this, photo](const QString &result) {
|
||||
if (!result.isEmpty()) {
|
||||
photo->full->pix().toImage().save(result, "JPG");
|
||||
}
|
||||
}));
|
||||
}
|
||||
|
||||
void ListWidget::saveDocumentToFile(DocumentData *document) {
|
||||
|
@ -900,50 +919,28 @@ void ListWidget::copyContextUrl() {
|
|||
}
|
||||
}
|
||||
|
||||
void ListWidget::showStickerPackInfo() {
|
||||
if (!App::contextItem()) return;
|
||||
|
||||
if (auto media = App::contextItem()->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
if (auto sticker = doc->sticker()) {
|
||||
if (sticker->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
App::main()->stickersBox(sticker->set);
|
||||
}
|
||||
}
|
||||
void ListWidget::showStickerPackInfo(not_null<DocumentData*> document) {
|
||||
if (auto sticker = document->sticker()) {
|
||||
if (sticker->set.type() != mtpc_inputStickerSetEmpty) {
|
||||
App::main()->stickersBox(sticker->set);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void ListWidget::cancelContextDownload() {
|
||||
if (auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get())) {
|
||||
lnkDocument->document()->cancel();
|
||||
} else if (auto item = App::contextItem()) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
doc->cancel();
|
||||
}
|
||||
}
|
||||
}
|
||||
void ListWidget::cancelContextDownload(not_null<DocumentData*> document) {
|
||||
document->cancel();
|
||||
}
|
||||
|
||||
void ListWidget::showContextInFolder() {
|
||||
QString filepath;
|
||||
if (auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.get())) {
|
||||
filepath = lnkDocument->document()->filepath(DocumentData::FilePathResolveChecked);
|
||||
} else if (auto item = App::contextItem()) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto doc = media->getDocument()) {
|
||||
filepath = doc->filepath(DocumentData::FilePathResolveChecked);
|
||||
}
|
||||
}
|
||||
}
|
||||
void ListWidget::showContextInFolder(not_null<DocumentData*> document) {
|
||||
const auto filepath = document->filepath(
|
||||
DocumentData::FilePathResolveChecked);
|
||||
if (!filepath.isEmpty()) {
|
||||
File::ShowInFolder(filepath);
|
||||
}
|
||||
}
|
||||
|
||||
void ListWidget::openContextGif() {
|
||||
if (auto item = App::contextItem()) {
|
||||
void ListWidget::openContextGif(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (auto media = item->getMedia()) {
|
||||
if (auto document = media->getDocument()) {
|
||||
Messenger::Instance().showDocument(document, item);
|
||||
|
@ -952,16 +949,20 @@ void ListWidget::openContextGif() {
|
|||
}
|
||||
}
|
||||
|
||||
void ListWidget::copyContextText() {
|
||||
auto item = App::contextItem();
|
||||
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
|
||||
return;
|
||||
void ListWidget::copyContextText(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
if (const auto media = item->getMedia()) {
|
||||
if (media->type() == MediaTypeSticker) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
setToClipboard(item->selectedText(FullSelection));
|
||||
}
|
||||
|
||||
setToClipboard(item->selectedText(FullSelection));
|
||||
}
|
||||
|
||||
void ListWidget::setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode) {
|
||||
void ListWidget::setToClipboard(
|
||||
const TextWithEntities &forClipboard,
|
||||
QClipboard::Mode mode) {
|
||||
if (auto data = MimeDataFromTextWithEntities(forClipboard)) {
|
||||
QApplication::clipboard()->setMimeData(data.release(), mode);
|
||||
}
|
||||
|
@ -996,8 +997,8 @@ void ListWidget::enterEventHook(QEvent *e) {
|
|||
}
|
||||
|
||||
void ListWidget::leaveEventHook(QEvent *e) {
|
||||
if (auto item = App::hoveredItem()) {
|
||||
repaintItem(viewForItem(item));
|
||||
if (const auto view = App::hoveredItem()) {
|
||||
repaintItem(view);
|
||||
App::hoveredItem(nullptr);
|
||||
}
|
||||
ClickHandler::clearActive();
|
||||
|
@ -1015,13 +1016,13 @@ void ListWidget::mouseActionStart(const QPoint &screenPos, Qt::MouseButton butto
|
|||
|
||||
ClickHandler::pressed();
|
||||
if (App::pressedItem() != App::hoveredItem()) {
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
repaintItem(App::pressedItem());
|
||||
App::pressedItem(App::hoveredItem());
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
repaintItem(App::pressedItem());
|
||||
}
|
||||
|
||||
_mouseAction = MouseAction::None;
|
||||
_mouseActionItem = viewForItem(App::mousedItem());
|
||||
_mouseActionItem = App::mousedItem();
|
||||
_dragStartPosition = mapPointToItem(mapFromGlobal(screenPos), _mouseActionItem);
|
||||
_pressWasInactive = _controller->window()->wasInactivePress();
|
||||
if (_pressWasInactive) _controller->window()->setInactivePress(false);
|
||||
|
@ -1102,8 +1103,8 @@ void ListWidget::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton butt
|
|||
if (_mouseAction == MouseAction::Dragging) {
|
||||
activated = nullptr;
|
||||
}
|
||||
if (App::pressedItem()) {
|
||||
repaintItem(viewForItem(App::pressedItem()));
|
||||
if (const auto view = App::pressedItem()) {
|
||||
repaintItem(view);
|
||||
App::pressedItem(nullptr);
|
||||
}
|
||||
|
||||
|
@ -1144,23 +1145,25 @@ void ListWidget::updateSelected() {
|
|||
const auto view = strictFindItemByY(point.y());
|
||||
const auto item = view ? view->data().get() : nullptr;
|
||||
if (view) {
|
||||
App::mousedItem(item);
|
||||
App::mousedItem(view);
|
||||
itemPoint = mapPointToItem(point, view);
|
||||
if (item->hasPoint(itemPoint)) {
|
||||
if (App::hoveredItem() != item) {
|
||||
repaintItem(viewForItem(App::hoveredItem()));
|
||||
App::hoveredItem(item);
|
||||
if (App::hoveredItem() != view) {
|
||||
repaintItem(App::hoveredItem());
|
||||
App::hoveredItem(view);
|
||||
repaintItem(view);
|
||||
}
|
||||
} else if (App::hoveredItem()) {
|
||||
repaintItem(viewForItem(App::hoveredItem()));
|
||||
} else if (const auto view = App::hoveredItem()) {
|
||||
repaintItem(view);
|
||||
App::hoveredItem(nullptr);
|
||||
}
|
||||
}
|
||||
|
||||
HistoryTextState dragState;
|
||||
ClickHandlerHost *lnkhost = nullptr;
|
||||
auto selectingText = (view == _mouseActionItem && item == App::hoveredItem() && _selectedItem);
|
||||
auto selectingText = _selectedItem
|
||||
&& (view == _mouseActionItem)
|
||||
&& (view == App::hoveredItem());
|
||||
if (view) {
|
||||
if (view != _mouseActionItem || (itemPoint - _dragStartPosition).manhattanLength() >= QApplication::startDragDistance()) {
|
||||
if (_mouseAction == MouseAction::PrepareDrag) {
|
||||
|
@ -1246,14 +1249,9 @@ void ListWidget::updateSelected() {
|
|||
}
|
||||
|
||||
// Voice message seek support.
|
||||
if (auto pressedItem = App::pressedLinkItem()) {
|
||||
if (!pressedItem->detached()) {
|
||||
// #TODO seek support
|
||||
//if (pressedItem->history() == _history) {
|
||||
// auto adjustedPoint = mapPointToItem(point, pressedItem);
|
||||
// pressedItem->updatePressed(adjustedPoint);
|
||||
//}
|
||||
}
|
||||
if (const auto pressedView = App::pressedLinkItem()) {
|
||||
auto adjustedPoint = mapPointToItem(point, pressedView);
|
||||
pressedView->data()->updatePressed(adjustedPoint);
|
||||
}
|
||||
|
||||
//if (_mouseAction == MouseAction::Selecting) {
|
||||
|
|
|
@ -160,12 +160,12 @@ private:
|
|||
void savePhotoToFile(PhotoData *photo);
|
||||
void saveDocumentToFile(DocumentData *document);
|
||||
void copyContextImage(PhotoData *photo);
|
||||
void showStickerPackInfo();
|
||||
void showStickerPackInfo(not_null<DocumentData*> document);
|
||||
void copyContextUrl();
|
||||
void cancelContextDownload();
|
||||
void showContextInFolder();
|
||||
void openContextGif();
|
||||
void copyContextText();
|
||||
void cancelContextDownload(not_null<DocumentData*> document);
|
||||
void showContextInFolder(not_null<DocumentData*> document);
|
||||
void openContextGif(FullMsgId itemId);
|
||||
void copyContextText(FullMsgId itemId);
|
||||
void copySelectedText();
|
||||
TextWithEntities getSelectedText() const;
|
||||
void setToClipboard(
|
||||
|
@ -254,8 +254,7 @@ private:
|
|||
bool _wasSelectedText = false; // was some text selected in current drag action
|
||||
Qt::CursorShape _cursor = style::cur_default;
|
||||
|
||||
// context menu
|
||||
Ui::PopupMenu *_menu = nullptr;
|
||||
base::unique_qptr<Ui::PopupMenu> _menu;
|
||||
|
||||
QPoint _trippleClickPoint;
|
||||
base::Timer _trippleClickTimer;
|
||||
|
|
|
@ -14,12 +14,56 @@ Message::Message(not_null<HistoryItem*> data, Context context)
|
|||
, _context(context) {
|
||||
}
|
||||
|
||||
MsgId Message::id() const {
|
||||
return _data->id;
|
||||
}
|
||||
|
||||
not_null<HistoryItem*> Message::data() const {
|
||||
return _data;
|
||||
}
|
||||
|
||||
void Message::attachToBlock(not_null<HistoryBlock*> block, int index) {
|
||||
Expects(!_data->isLogEntry());
|
||||
Expects(_block == nullptr);
|
||||
Expects(_indexInBlock < 0);
|
||||
Expects(index >= 0);
|
||||
|
||||
_block = block;
|
||||
_indexInBlock = index;
|
||||
_data->setMainView(this);
|
||||
_data->setPendingResize();
|
||||
}
|
||||
|
||||
void Message::removeFromBlock() {
|
||||
Expects(_block != nullptr);
|
||||
|
||||
_block->remove(this);
|
||||
}
|
||||
|
||||
Message *Message::previousInBlocks() const {
|
||||
if (_block && _indexInBlock >= 0) {
|
||||
if (_indexInBlock > 0) {
|
||||
return _block->messages[_indexInBlock - 1].get();
|
||||
}
|
||||
if (auto previous = _block->previousBlock()) {
|
||||
Assert(!previous->messages.empty());
|
||||
return previous->messages.back().get();
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
Message *Message::nextInBlocks() const {
|
||||
if (_block && _indexInBlock >= 0) {
|
||||
if (_indexInBlock + 1 < _block->messages.size()) {
|
||||
return _block->messages[_indexInBlock + 1].get();
|
||||
}
|
||||
if (auto next = _block->nextBlock()) {
|
||||
Assert(!next->messages.empty());
|
||||
return next->messages.front().get();
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
Message::~Message() {
|
||||
App::messageViewDestroyed(this);
|
||||
}
|
||||
|
||||
} // namespace HistoryView
|
||||
|
|
|
@ -23,7 +23,6 @@ class Message
|
|||
public:
|
||||
Message(not_null<HistoryItem*> data, Context context);
|
||||
|
||||
MsgId id() const;
|
||||
not_null<HistoryItem*> data() const;
|
||||
|
||||
int y() const {
|
||||
|
@ -33,11 +32,39 @@ public:
|
|||
_y = y;
|
||||
}
|
||||
|
||||
HistoryBlock *block() {
|
||||
return _block;
|
||||
}
|
||||
const HistoryBlock *block() const {
|
||||
return _block;
|
||||
}
|
||||
void attachToBlock(not_null<HistoryBlock*> block, int index);
|
||||
void removeFromBlock();
|
||||
void setIndexInBlock(int index) {
|
||||
Expects(_block != nullptr);
|
||||
Expects(index >= 0);
|
||||
|
||||
_indexInBlock = index;
|
||||
}
|
||||
int indexInBlock() const {
|
||||
Expects((_indexInBlock >= 0) == (_block != nullptr));
|
||||
Expects((_block == nullptr) || (_block->messages[_indexInBlock].get() == this));
|
||||
|
||||
return _indexInBlock;
|
||||
}
|
||||
Message *previousInBlocks() const;
|
||||
Message *nextInBlocks() const;
|
||||
|
||||
~Message();
|
||||
|
||||
private:
|
||||
const not_null<HistoryItem*> _data;
|
||||
int _y = 0;
|
||||
Context _context;
|
||||
|
||||
HistoryBlock *_block = nullptr;
|
||||
int _indexInBlock = -1;
|
||||
|
||||
};
|
||||
|
||||
} // namespace HistoryView
|
||||
|
|
|
@ -596,17 +596,19 @@ bool MainWidget::setForwardDraft(PeerId peerId, ForwardWhatMessages what) {
|
|||
return _history->getSelectedItems();
|
||||
}
|
||||
auto item = (HistoryItem*)nullptr;
|
||||
if (what == ForwardContextMessage) {
|
||||
item = App::contextItem();
|
||||
} else if (what == ForwardPressedMessage) {
|
||||
item = App::pressedItem();
|
||||
if (what == ForwardPressedMessage) {
|
||||
item = App::pressedItem()
|
||||
? App::pressedItem()->data().get()
|
||||
: nullptr;
|
||||
if (const auto group = item ? item->getFullGroup() : nullptr) {
|
||||
if (item->id > 0) {
|
||||
return Auth().data().groupToIds(group);
|
||||
}
|
||||
}
|
||||
} else if (what == ForwardPressedLinkMessage) {
|
||||
item = App::pressedLinkItem();
|
||||
item = App::pressedLinkItem()
|
||||
? App::pressedLinkItem()->data().get()
|
||||
: nullptr;
|
||||
}
|
||||
if (item && item->toHistoryMessage() && item->id > 0) {
|
||||
return { 1, item->fullId() };
|
||||
|
@ -677,8 +679,7 @@ bool MainWidget::shareUrl(
|
|||
void MainWidget::replyToItem(not_null<HistoryItem*> item) {
|
||||
if (_history->peer() == item->history()->peer
|
||||
|| _history->peer() == item->history()->peer->migrateTo()) {
|
||||
App::contextItem(item);
|
||||
_history->onReplyToMessage();
|
||||
_history->replyToMessage(item);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -932,24 +933,20 @@ void MainWidget::showSendPathsLayer() {
|
|||
hiderLayer(object_ptr<HistoryHider>(this));
|
||||
}
|
||||
|
||||
void MainWidget::deleteLayer(int selectedCount) {
|
||||
if (selectedCount) {
|
||||
auto selected = _history->getSelectedItems();
|
||||
if (!selected.empty()) {
|
||||
Ui::show(Box<DeleteMessagesBox>(std::move(selected)));
|
||||
}
|
||||
} else if (const auto item = App::contextItem()) {
|
||||
void MainWidget::deleteLayer(MessageIdsList &&items) {
|
||||
if (!items.empty()) {
|
||||
Ui::show(Box<DeleteMessagesBox>(std::move(items)));
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::deleteLayer(FullMsgId itemId) {
|
||||
if (const auto item = App::histItemById(itemId)) {
|
||||
const auto suggestModerateActions = true;
|
||||
Ui::show(Box<DeleteMessagesBox>(item, suggestModerateActions));
|
||||
}
|
||||
}
|
||||
|
||||
void MainWidget::cancelUploadLayer() {
|
||||
auto item = App::contextItem();
|
||||
if (!item) {
|
||||
return;
|
||||
}
|
||||
|
||||
void MainWidget::cancelUploadLayer(not_null<HistoryItem*> item) {
|
||||
const auto itemId = item->fullId();
|
||||
Auth().uploader().pause(itemId);
|
||||
Ui::show(Box<ConfirmBox>(lang(lng_selected_cancel_sure_this), lang(lng_selected_upload_stop), lang(lng_continue), base::lambda_guarded(this, [=] {
|
||||
|
@ -2111,7 +2108,7 @@ void MainWidget::updateBotKeyboard(History *h) {
|
|||
_history->updateBotKeyboard(h);
|
||||
}
|
||||
|
||||
void MainWidget::pushReplyReturn(HistoryItem *item) {
|
||||
void MainWidget::pushReplyReturn(not_null<HistoryItem*> item) {
|
||||
_history->pushReplyReturn(item);
|
||||
}
|
||||
|
||||
|
@ -4931,7 +4928,7 @@ void MainWidget::feedUpdate(const MTPUpdate &update) {
|
|||
const auto newId = d.vid.v;
|
||||
if (const auto local = App::histItemById(fullId)) {
|
||||
const auto existing = App::histItemById(channel, newId);
|
||||
if (existing && local->detached()) {
|
||||
if (existing && !local->mainView()) {
|
||||
const auto history = local->history();
|
||||
const auto wasLast = (history->lastMsg == local);
|
||||
local->destroy();
|
||||
|
|
|
@ -168,8 +168,9 @@ public:
|
|||
|
||||
void showForwardLayer(MessageIdsList &&items);
|
||||
void showSendPathsLayer();
|
||||
void deleteLayer(int selectedCount = 0); // 0 - context item
|
||||
void cancelUploadLayer();
|
||||
void deleteLayer(MessageIdsList &&items);
|
||||
void deleteLayer(FullMsgId itemId);
|
||||
void cancelUploadLayer(not_null<HistoryItem*> item);
|
||||
void shareUrlLayer(const QString &url, const QString &text);
|
||||
void inlineSwitchLayer(const QString &botAndQuery);
|
||||
void hiderLayer(object_ptr<HistoryHider> h);
|
||||
|
@ -272,7 +273,7 @@ public:
|
|||
void messageDataReceived(ChannelData *channel, MsgId msgId);
|
||||
void updateBotKeyboard(History *h);
|
||||
|
||||
void pushReplyReturn(HistoryItem *item);
|
||||
void pushReplyReturn(not_null<HistoryItem*> item);
|
||||
|
||||
void cancelForwarding(not_null<History*> history);
|
||||
void finishForwarding(not_null<History*> history);
|
||||
|
|
|
@ -1003,8 +1003,10 @@ void MediaView::onForward() {
|
|||
|
||||
void MediaView::onDelete() {
|
||||
close();
|
||||
auto deletingPeerPhoto = [this]() {
|
||||
if (!_msgid) return true;
|
||||
const auto deletingPeerPhoto = [this] {
|
||||
if (!_msgid) {
|
||||
return true;
|
||||
}
|
||||
if (_photo && _history) {
|
||||
if (_history->peer->userpicPhotoId() == _photo->id) {
|
||||
return _firstOpenedPeerPhoto;
|
||||
|
@ -1015,9 +1017,8 @@ void MediaView::onDelete() {
|
|||
|
||||
if (deletingPeerPhoto()) {
|
||||
App::main()->deletePhotoLayer(_photo);
|
||||
} else if (auto item = App::histItemById(_msgid)) {
|
||||
App::contextItem(item);
|
||||
App::main()->deleteLayer();
|
||||
} else {
|
||||
App::main()->deleteLayer(_msgid);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -120,7 +120,8 @@ ItemBase::ItemBase(not_null<HistoryItem*> parent) : _parent(parent) {
|
|||
void ItemBase::clickHandlerActiveChanged(
|
||||
const ClickHandlerPtr &action,
|
||||
bool active) {
|
||||
App::hoveredLinkItem(active ? _parent.get() : nullptr);
|
||||
// #TODO hoveredLinkItem
|
||||
// App::hoveredLinkItem(active ? _parent.get() : nullptr);
|
||||
Auth().data().requestItemRepaint(_parent);
|
||||
if (_check) {
|
||||
_check->setActive(active);
|
||||
|
@ -130,7 +131,8 @@ void ItemBase::clickHandlerActiveChanged(
|
|||
void ItemBase::clickHandlerPressedChanged(
|
||||
const ClickHandlerPtr &action,
|
||||
bool pressed) {
|
||||
App::pressedLinkItem(pressed ? _parent.get() : nullptr);
|
||||
// #TODO pressedLinkItem
|
||||
// App::pressedLinkItem(pressed ? _parent.get() : nullptr);
|
||||
Auth().data().requestItemRepaint(_parent);
|
||||
if (_check) {
|
||||
_check->setPressed(pressed);
|
||||
|
@ -172,18 +174,25 @@ ItemBase::~ItemBase() = default;
|
|||
|
||||
void RadialProgressItem::setDocumentLinks(
|
||||
not_null<DocumentData*> document) {
|
||||
auto createSaveHandler = [](
|
||||
not_null<DocumentData*> document
|
||||
) -> ClickHandlerPtr {
|
||||
const auto context = parent()->fullId();
|
||||
const auto createSaveHandler = [&]() -> ClickHandlerPtr {
|
||||
if (document->isVoiceMessage()) {
|
||||
return std::make_shared<DocumentOpenClickHandler>(document);
|
||||
return std::make_shared<DocumentOpenClickHandler>(
|
||||
document,
|
||||
context);
|
||||
}
|
||||
return std::make_shared<DocumentSaveClickHandler>(document);
|
||||
return std::make_shared<DocumentSaveClickHandler>(
|
||||
document,
|
||||
context);
|
||||
};
|
||||
setLinks(
|
||||
std::make_shared<DocumentOpenClickHandler>(document),
|
||||
createSaveHandler(document),
|
||||
std::make_shared<DocumentCancelClickHandler>(document));
|
||||
std::make_shared<DocumentOpenClickHandler>(
|
||||
document,
|
||||
context),
|
||||
createSaveHandler(),
|
||||
std::make_shared<DocumentCancelClickHandler>(
|
||||
document,
|
||||
context));
|
||||
}
|
||||
|
||||
void RadialProgressItem::clickHandlerActiveChanged(const ClickHandlerPtr &action, bool active) {
|
||||
|
@ -539,7 +548,7 @@ Voice::Voice(
|
|||
const style::OverviewFileLayout &st)
|
||||
: RadialProgressItem(parent)
|
||||
, _data(voice)
|
||||
, _namel(std::make_shared<DocumentOpenClickHandler>(_data))
|
||||
, _namel(std::make_shared<DocumentOpenClickHandler>(_data, parent->fullId()))
|
||||
, _st(st) {
|
||||
AddComponents(Info::Bit());
|
||||
|
||||
|
@ -784,7 +793,7 @@ Document::Document(
|
|||
: RadialProgressItem(parent)
|
||||
, _data(document)
|
||||
, _msgl(goToMessageClickHandler(parent))
|
||||
, _namel(std::make_shared<DocumentOpenClickHandler>(_data))
|
||||
, _namel(std::make_shared<DocumentOpenClickHandler>(_data, parent->fullId()))
|
||||
, _st(st)
|
||||
, _date(langDateTime(date(_data->date)))
|
||||
, _datew(st::normalFont->width(_date))
|
||||
|
@ -1205,7 +1214,9 @@ Link::Link(
|
|||
if (_page) {
|
||||
mainUrl = _page->url;
|
||||
if (_page->document) {
|
||||
_photol = std::make_shared<DocumentOpenClickHandler>(_page->document);
|
||||
_photol = std::make_shared<DocumentOpenClickHandler>(
|
||||
_page->document,
|
||||
parent->fullId());
|
||||
} else if (_page->photo) {
|
||||
if (_page->type == WebPageProfile || _page->type == WebPageVideo) {
|
||||
_photol = std::make_shared<UrlClickHandler>(_page->url);
|
||||
|
|
Loading…
Reference in New Issue