Enable distinct selecting of grouped media.

This commit is contained in:
John Preston 2017-12-15 19:25:47 +03:00
parent 4c9931ab02
commit 537400d8b2
22 changed files with 880 additions and 306 deletions

View File

@ -224,7 +224,10 @@ QString documentSaveFilename(const DocumentData *data, bool forceSavingAs = fals
return saveFileName(caption, filter, prefix, name, forceSavingAs, dir); return saveFileName(caption, filter, prefix, name, forceSavingAs, dir);
} }
void DocumentOpenClickHandler::doOpen(DocumentData *data, HistoryItem *context, ActionOnLoad action) { void DocumentOpenClickHandler::doOpen(
not_null<DocumentData*> data,
HistoryItem *context,
ActionOnLoad action) {
if (!data->date) return; if (!data->date) return;
auto msgId = context ? context->fullId() : FullMsgId(); auto msgId = context ? context->fullId() : FullMsgId();
@ -329,9 +332,13 @@ void DocumentOpenClickHandler::doOpen(DocumentData *data, HistoryItem *context,
} }
void DocumentOpenClickHandler::onClickImpl() const { void DocumentOpenClickHandler::onClickImpl() const {
const auto item = App::hoveredLinkItem() const auto item = context()
? App::histItemById(context())
: App::hoveredLinkItem()
? App::hoveredLinkItem() ? App::hoveredLinkItem()
: (App::contextItem() ? App::contextItem() : nullptr); : App::contextItem()
? App::contextItem()
: nullptr;
const auto action = document()->isVoiceMessage() const auto action = document()->isVoiceMessage()
? ActionOnLoadNone ? ActionOnLoadNone
: ActionOnLoadOpen; : ActionOnLoadOpen;
@ -339,13 +346,19 @@ void DocumentOpenClickHandler::onClickImpl() const {
} }
void GifOpenClickHandler::onClickImpl() const { void GifOpenClickHandler::onClickImpl() const {
const auto item = App::hoveredLinkItem() const auto item = context()
? App::histItemById(context())
: App::hoveredLinkItem()
? App::hoveredLinkItem() ? App::hoveredLinkItem()
: (App::contextItem() ? App::contextItem() : nullptr); : App::contextItem()
? App::contextItem()
: nullptr;
doOpen(document(), item, ActionOnLoadPlayInline); doOpen(document(), item, ActionOnLoadPlayInline);
} }
void DocumentSaveClickHandler::doSave(DocumentData *data, bool forceSavingAs) { void DocumentSaveClickHandler::doSave(
not_null<DocumentData*> data,
bool forceSavingAs) {
if (!data->date) return; if (!data->date) return;
auto filepath = data->filepath(DocumentData::FilePathResolveSaveFromDataSilent, forceSavingAs); auto filepath = data->filepath(DocumentData::FilePathResolveSaveFromDataSilent, forceSavingAs);

View File

@ -311,15 +311,22 @@ QByteArray documentWaveformEncode5bit(const VoiceWaveform &waveform);
class DocumentClickHandler : public LeftButtonClickHandler { class DocumentClickHandler : public LeftButtonClickHandler {
public: public:
DocumentClickHandler(DocumentData *document) DocumentClickHandler(
: _document(document) { not_null<DocumentData*> document,
FullMsgId context = FullMsgId())
: _document(document)
, _context(context) {
} }
DocumentData *document() const { not_null<DocumentData*> document() const {
return _document; return _document;
} }
FullMsgId context() const {
return _context;
}
private: private:
DocumentData *_document; not_null<DocumentData*> _document;
FullMsgId _context;
}; };
@ -327,7 +334,7 @@ class DocumentSaveClickHandler : public DocumentClickHandler {
public: public:
using DocumentClickHandler::DocumentClickHandler; using DocumentClickHandler::DocumentClickHandler;
static void doSave( static void doSave(
DocumentData *document, not_null<DocumentData*> document,
bool forceSavingAs = false); bool forceSavingAs = false);
protected: protected:
@ -339,7 +346,7 @@ class DocumentOpenClickHandler : public DocumentClickHandler {
public: public:
using DocumentClickHandler::DocumentClickHandler; using DocumentClickHandler::DocumentClickHandler;
static void doOpen( static void doOpen(
DocumentData *document, not_null<DocumentData*> document,
HistoryItem *context, HistoryItem *context,
ActionOnLoad action = ActionOnLoadOpen); ActionOnLoad action = ActionOnLoadOpen);

View File

@ -121,7 +121,7 @@ ImagePtr PhotoData::makeReplyPreview() {
} }
void PhotoOpenClickHandler::onClickImpl() const { void PhotoOpenClickHandler::onClickImpl() const {
Messenger::Instance().showPhoto(this, App::hoveredLinkItem() ? App::hoveredLinkItem() : App::contextItem()); Messenger::Instance().showPhoto(this);
} }
void PhotoSaveClickHandler::onClickImpl() const { void PhotoSaveClickHandler::onClickImpl() const {
@ -136,13 +136,9 @@ void PhotoCancelClickHandler::onClickImpl() const {
if (!data->date) return; if (!data->date) return;
if (data->uploading()) { if (data->uploading()) {
if (auto item = App::hoveredLinkItem() ? App::hoveredLinkItem() : (App::contextItem() ? App::contextItem() : nullptr)) { if (const auto item = App::histItemById(context())) {
if (auto media = item->getMedia()) { App::contextItem(item);
if (media->type() == MediaTypePhoto && static_cast<HistoryPhoto*>(media)->photo() == data) { App::main()->cancelUploadLayer();
App::contextItem(item);
App::main()->cancelUploadLayer();
}
}
} }
} else { } else {
data->cancel(); data->cancel();

View File

@ -74,8 +74,11 @@ class PhotoClickHandler : public LeftButtonClickHandler {
public: public:
PhotoClickHandler( PhotoClickHandler(
not_null<PhotoData*> photo, not_null<PhotoData*> photo,
FullMsgId context = FullMsgId(),
PeerData *peer = nullptr) PeerData *peer = nullptr)
: _photo(photo), _peer(peer) { : _photo(photo)
, _context(context)
, _peer(peer) {
} }
not_null<PhotoData*> photo() const { not_null<PhotoData*> photo() const {
return _photo; return _photo;
@ -83,10 +86,14 @@ public:
PeerData *peer() const { PeerData *peer() const {
return _peer; return _peer;
} }
FullMsgId context() const {
return _context;
}
private: private:
not_null<PhotoData*> _photo; not_null<PhotoData*> _photo;
PeerData *_peer; FullMsgId _context;
PeerData *_peer = nullptr;
}; };

View File

@ -363,6 +363,55 @@ void HistoryInner::enumerateDates(Method method) {
enumerateItems<EnumItemsDirection::BottomToTop>(dateCallback); enumerateItems<EnumItemsDirection::BottomToTop>(dateCallback);
} }
TextSelection HistoryInner::itemRenderSelection(
not_null<HistoryItem*> item,
int selfromy,
int seltoy) const {
Expects(!item->detached());
const auto y = item->block()->y() + item->y();
if (y >= selfromy && y < seltoy) {
if (_dragSelecting && !item->serviceMsg() && item->id > 0) {
return FullSelection;
}
} else if (!_selected.empty()) {
const auto itemSelection = [&](not_null<HistoryItem*> item) {
auto i = _selected.find(item);
if (i != _selected.end()) {
return i->second;
}
return TextSelection();
};
const auto group = item->Get<HistoryMessageGroup>();
if (group) {
if (group->leader != item) {
return TextSelection();
}
auto result = TextSelection();
auto allFullSelected = true;
const auto count = int(group->others.size());
for (auto i = 0; i != count; ++i) {
if (itemSelection(group->others[i]) == FullSelection) {
result = AddGroupItemSelection(result, i);
} else {
allFullSelected = false;
}
}
const auto leaderSelection = itemSelection(item);
if (leaderSelection == FullSelection) {
return allFullSelected
? FullSelection
: AddGroupItemSelection(result, count);
} else if (leaderSelection != TextSelection()) {
return leaderSelection;
}
return result;
}
return itemSelection(item);
}
return TextSelection();
}
void HistoryInner::paintEvent(QPaintEvent *e) { void HistoryInner::paintEvent(QPaintEvent *e) {
if (Ui::skipPaintEvent(this, e)) { if (Ui::skipPaintEvent(this, e)) {
return; return;
@ -399,9 +448,6 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
adjustCurrent(clip.top()); adjustCurrent(clip.top());
auto selEnd = _selected.cend();
auto hasSel = !_selected.empty();
auto drawToY = clip.y() + clip.height(); auto drawToY = clip.y() + clip.height();
auto selfromy = itemTop(_dragSelFrom); auto selfromy = itemTop(_dragSelFrom);
@ -425,18 +471,11 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
p.save(); p.save();
p.translate(0, y); p.translate(0, y);
if (clip.y() < y + item->height()) while (y < drawToY) { if (clip.y() < y + item->height()) while (y < drawToY) {
TextSelection sel; const auto selection = itemRenderSelection(
if (y >= selfromy && y < seltoy) { item,
if (_dragSelecting && !item->serviceMsg() && item->id > 0) { selfromy - mtop,
sel = FullSelection; seltoy - mtop);
} item->draw(p, clip.translated(0, -y), selection, ms);
} else if (hasSel) {
auto i = _selected.find(item);
if (i != selEnd) {
sel = i->second;
}
}
item->draw(p, clip.translated(0, -y), sel, ms);
if (item->hasViews()) { if (item->hasViews()) {
App::main()->scheduleViewIncrement(item); App::main()->scheduleViewIncrement(item);
@ -469,25 +508,18 @@ void HistoryInner::paintEvent(QPaintEvent *e) {
auto iItem = (_curHistory == _history ? _curItem : 0); auto iItem = (_curHistory == _history ? _curItem : 0);
auto item = block->items[iItem]; auto item = block->items[iItem];
auto historyRect = clip.intersected(QRect(0, hdrawtop, width(), clip.top() + clip.height())); auto hclip = clip.intersected(QRect(0, hdrawtop, width(), clip.top() + clip.height()));
auto y = htop + block->y() + item->y(); auto y = htop + block->y() + item->y();
p.save(); p.save();
p.translate(0, y); p.translate(0, y);
while (y < drawToY) { while (y < drawToY) {
auto h = item->height(); auto h = item->height();
if (historyRect.y() < y + h && hdrawtop < y + h) { if (hclip.y() < y + h && hdrawtop < y + h) {
TextSelection sel; const auto selection = itemRenderSelection(
if (y >= selfromy && y < seltoy) { item,
if (_dragSelecting && !item->serviceMsg() && item->id > 0) { selfromy - htop,
sel = FullSelection; seltoy - htop);
} item->draw(p, hclip.translated(0, -y), selection, ms);
} else if (hasSel) {
auto i = _selected.find(item);
if (i != selEnd) {
sel = i->second;
}
}
item->draw(p, historyRect.translated(0, -y), sel, ms);
if (item->hasViews()) { if (item->hasViews()) {
App::main()->scheduleViewIncrement(item); App::main()->scheduleViewIncrement(item);
@ -830,7 +862,9 @@ void HistoryInner::mouseActionStart(const QPoint &screenPos, Qt::MouseButton but
_mouseAction = MouseAction::PrepareDrag; _mouseAction = MouseAction::PrepareDrag;
} else if (!_selected.empty()) { } else if (!_selected.empty()) {
if (_selected.cbegin()->second == FullSelection) { if (_selected.cbegin()->second == FullSelection) {
if (_selected.find(_mouseActionItem) != _selected.cend() && App::hoveredItem()) { if (_dragStateItem
&& _selected.find(_dragStateItem) != _selected.cend()
&& App::hoveredItem()) {
_mouseAction = MouseAction::PrepareDrag; // start items drag _mouseAction = MouseAction::PrepareDrag; // start items drag
} else if (!_pressWasInactive) { } else if (!_pressWasInactive) {
_mouseAction = MouseAction::PrepareSelect; // start items select _mouseAction = MouseAction::PrepareSelect; // start items select
@ -915,6 +949,7 @@ void HistoryInner::mouseActionStart(const QPoint &screenPos, Qt::MouseButton but
void HistoryInner::mouseActionCancel() { void HistoryInner::mouseActionCancel() {
_mouseActionItem = nullptr; _mouseActionItem = nullptr;
_dragStateItem = nullptr;
_mouseAction = MouseAction::None; _mouseAction = MouseAction::None;
_dragStartPosition = QPoint(0, 0); _dragStartPosition = QPoint(0, 0);
_dragSelFrom = _dragSelTo = nullptr; _dragSelFrom = _dragSelTo = nullptr;
@ -928,7 +963,8 @@ void HistoryInner::performDrag() {
bool uponSelected = false; bool uponSelected = false;
if (_mouseActionItem) { if (_mouseActionItem) {
if (!_selected.empty() && _selected.cbegin()->second == FullSelection) { if (!_selected.empty() && _selected.cbegin()->second == FullSelection) {
uponSelected = (_selected.find(_mouseActionItem) != _selected.cend()); uponSelected = _dragStateItem
&& (_selected.find(_dragStateItem) != _selected.cend());
} else { } else {
HistoryStateRequest request; HistoryStateRequest request;
request.flags |= Text::StateRequest::Flag::LookupSymbol; request.flags |= Text::StateRequest::Flag::LookupSymbol;
@ -986,7 +1022,7 @@ void HistoryInner::performDrag() {
forwardMimeType = qsl("application/x-td-forward-pressed"); forwardMimeType = qsl("application/x-td-forward-pressed");
} }
} }
if (auto pressedLnkItem = App::pressedLinkItem()) { if (const auto pressedLnkItem = _dragStateItem) {
if ((pressedMedia = pressedLnkItem->getMedia())) { if ((pressedMedia = pressedLnkItem->getMedia())) {
if (forwardMimeType.isEmpty() && pressedMedia->dragItemByHandler(pressedHandler)) { if (forwardMimeType.isEmpty() && pressedMedia->dragItemByHandler(pressedHandler)) {
forwardMimeType = qsl("application/x-td-forward-pressed-link"); forwardMimeType = qsl("application/x-td-forward-pressed-link");
@ -1029,6 +1065,9 @@ void HistoryInner::itemRemoved(not_null<const HistoryItem*> item) {
if (_mouseActionItem == item) { if (_mouseActionItem == item) {
mouseActionCancel(); mouseActionCancel();
} }
if (_dragStateItem == item) {
_dragStateItem = nullptr;
}
if (_dragSelFrom == item || _dragSelTo == item) { if (_dragSelFrom == item || _dragSelTo == item) {
_dragSelFrom = 0; _dragSelFrom = 0;
@ -1041,15 +1080,14 @@ void HistoryInner::itemRemoved(not_null<const HistoryItem*> item) {
void HistoryInner::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton button) { void HistoryInner::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton button) {
mouseActionUpdate(screenPos); mouseActionUpdate(screenPos);
auto pressedLinkItem = App::pressedLinkItem();
auto activated = ClickHandler::unpressed(); auto activated = ClickHandler::unpressed();
if (_mouseAction == MouseAction::Dragging) { if (_mouseAction == MouseAction::Dragging) {
activated.clear(); activated.clear();
} else if (auto pressed = pressedLinkItem) { } else if (_mouseActionItem) {
// if we are in selecting items mode perhaps we want to // if we are in selecting items mode perhaps we want to
// toggle selection instead of activating the pressed link // toggle selection instead of activating the pressed link
if (_mouseAction == MouseAction::PrepareDrag && !_pressWasInactive && !_selected.empty() && _selected.cbegin()->second == FullSelection && button != Qt::RightButton) { if (_mouseAction == MouseAction::PrepareDrag && !_pressWasInactive && !_selected.empty() && _selected.cbegin()->second == FullSelection && button != Qt::RightButton) {
if (auto media = pressed->getMedia()) { if (auto media = _mouseActionItem->getMedia()) {
if (media->toggleSelectionByHandlerClick(activated)) { if (media->toggleSelectionByHandlerClick(activated)) {
activated.clear(); activated.clear();
} }
@ -1068,29 +1106,30 @@ void HistoryInner::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton bu
App::activateClickHandler(activated, button); App::activateClickHandler(activated, button);
return; return;
} }
if (_mouseAction == MouseAction::PrepareSelect && !_pressWasInactive && !_selected.empty() && _selected.cbegin()->second == FullSelection) { if ((_mouseAction == MouseAction::PrepareSelect)
auto i = _selected.find(_mouseActionItem); && !_pressWasInactive
if (i == _selected.cend()) { && !_selected.empty()
if (!_mouseActionItem->serviceMsg() && (_selected.cbegin()->second == FullSelection)) {
&& IsServerMsgId(_mouseActionItem->id) changeDragSelection(
&& _selected.size() < MaxSelectedItems) { &_selected,
if (!_selected.empty() && _selected.cbegin()->second != FullSelection) { _mouseActionItem,
_selected.clear(); SelectAction::Invert);
}
_selected.emplace(_mouseActionItem, FullSelection);
}
} else {
_selected.erase(i);
}
repaintItem(_mouseActionItem); repaintItem(_mouseActionItem);
} else if (_mouseAction == MouseAction::PrepareDrag && !_pressWasInactive && button != Qt::RightButton) { } else if ((_mouseAction == MouseAction::PrepareDrag)
auto i = _selected.find(_mouseActionItem); && !_pressWasInactive
&& _dragStateItem
&& (button != Qt::RightButton)) {
auto i = _selected.find(_dragStateItem);
if (i != _selected.cend() && i->second == FullSelection) { if (i != _selected.cend() && i->second == FullSelection) {
_selected.erase(i); _selected.erase(i);
repaintItem(_mouseActionItem); repaintItem(_mouseActionItem);
} else if (i == _selected.cend() && !_mouseActionItem->serviceMsg() && _mouseActionItem->id > 0 && !_selected.empty() && _selected.cbegin()->second == FullSelection) { } else if ((i == _selected.cend())
&& !_dragStateItem->serviceMsg()
&& (_dragStateItem->id > 0)
&& !_selected.empty()
&& _selected.cbegin()->second == FullSelection) {
if (_selected.size() < MaxSelectedItems) { if (_selected.size() < MaxSelectedItems) {
_selected.emplace(_mouseActionItem, FullSelection); _selected.emplace(_dragStateItem, FullSelection);
repaintItem(_mouseActionItem); repaintItem(_mouseActionItem);
} }
} else { } else {
@ -1117,7 +1156,8 @@ void HistoryInner::mouseActionFinish(const QPoint &screenPos, Qt::MouseButton bu
#if defined Q_OS_LINUX32 || defined Q_OS_LINUX64 #if defined Q_OS_LINUX32 || defined Q_OS_LINUX64
if (!_selected.empty() && _selected.cbegin()->second != FullSelection) { if (!_selected.empty() && _selected.cbegin()->second != FullSelection) {
setToClipboard(_selected.cbegin()->first->selectedText(_selected.cbegin()->second), QClipboard::Selection); const auto [item, selection] = *_selected.cbegin();
setToClipboard(item->selectedText(selection), QClipboard::Selection);
} }
#endif // Q_OS_LINUX32 || Q_OS_LINUX64 #endif // Q_OS_LINUX32 || Q_OS_LINUX64
} }
@ -1206,9 +1246,9 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
_menu = new Ui::PopupMenu(nullptr); _menu = new Ui::PopupMenu(nullptr);
_contextMenuLink = ClickHandler::getActive(); _contextMenuLink = ClickHandler::getActive();
HistoryItem *item = App::hoveredItem() ? App::hoveredItem() : App::hoveredLinkItem(); auto item = App::hoveredItem() ? App::hoveredItem() : App::hoveredLinkItem();
PhotoClickHandler *lnkPhoto = dynamic_cast<PhotoClickHandler*>(_contextMenuLink.data()); auto lnkPhoto = dynamic_cast<PhotoClickHandler*>(_contextMenuLink.data());
DocumentClickHandler *lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.data()); auto lnkDocument = dynamic_cast<DocumentClickHandler*>(_contextMenuLink.data());
auto lnkIsVideo = lnkDocument ? lnkDocument->document()->isVideoFile() : false; auto lnkIsVideo = lnkDocument ? lnkDocument->document()->isVideoFile() : false;
auto lnkIsVoice = lnkDocument ? lnkDocument->document()->isVoiceMessage() : false; auto lnkIsVoice = lnkDocument ? lnkDocument->document()->isVoiceMessage() : false;
auto lnkIsAudio = lnkDocument ? lnkDocument->document()->isAudioFile() : false; auto lnkIsAudio = lnkDocument ? lnkDocument->document()->isAudioFile() : false;
@ -1279,7 +1319,9 @@ void HistoryInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) {
} }
} }
if (App::hoveredLinkItem()->id > 0 && !App::hoveredLinkItem()->serviceMsg()) { if (App::hoveredLinkItem()->id > 0 && !App::hoveredLinkItem()->serviceMsg()) {
_menu->addAction(lang(lng_context_select_msg), _widget, SLOT(selectMessage()))->setEnabled(true); _menu->addAction(lang(lng_context_select_msg), base::lambda_guarded(this, [this] {
// TODO
}))->setEnabled(true);
} }
App::contextItem(App::hoveredLinkItem()); App::contextItem(App::hoveredLinkItem());
} }
@ -1513,7 +1555,7 @@ void HistoryInner::saveContextGif() {
} }
void HistoryInner::copyContextText() { void HistoryInner::copyContextText() {
auto item = App::contextItem(); const auto item = App::contextItem();
if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) { if (!item || (item->getMedia() && item->getMedia()->type() == MediaTypeSticker)) {
return; return;
} }
@ -1532,24 +1574,24 @@ void HistoryInner::resizeEvent(QResizeEvent *e) {
} }
TextWithEntities HistoryInner::getSelectedText() const { TextWithEntities HistoryInner::getSelectedText() const {
SelectedItems sel = _selected; auto selected = _selected;
if (_mouseAction == MouseAction::Selecting && _dragSelFrom && _dragSelTo) { if (_mouseAction == MouseAction::Selecting && _dragSelFrom && _dragSelTo) {
applyDragSelection(&sel); applyDragSelection(&selected);
} }
if (sel.empty()) { if (selected.empty()) {
return TextWithEntities(); return TextWithEntities();
} }
if (sel.cbegin()->second != FullSelection) { if (selected.cbegin()->second != FullSelection) {
return sel.cbegin()->first->selectedText(sel.cbegin()->second); const auto [item, selection] = *selected.cbegin();
return item->selectedText(selection);
} }
int fullSize = 0; int fullSize = 0;
QString timeFormat(qsl(", [dd.MM.yy hh:mm]\n")); QString timeFormat(qsl(", [dd.MM.yy hh:mm]\n"));
QMap<int, TextWithEntities> texts; QMap<int, TextWithEntities> texts;
for (auto &selected : sel) { for (const auto [item, selection] : selected) {
auto item = selected.first;
if (item->detached()) continue; if (item->detached()) continue;
auto time = item->date.toString(timeFormat); auto time = item->date.toString(timeFormat);
@ -2007,10 +2049,11 @@ MessageIdsList HistoryInner::getSelectedItems() const {
return result; return result;
} }
void HistoryInner::selectItem(HistoryItem *item) { void HistoryInner::selectItem(not_null<HistoryItem*> item) {
if (!_selected.empty() && _selected.cbegin()->second != FullSelection) { if (!_selected.empty() && _selected.cbegin()->second != FullSelection) {
_selected.clear(); _selected.clear();
} else if (_selected.size() == MaxSelectedItems && _selected.find(item) == _selected.cend()) { } else if (_selected.size() == MaxSelectedItems
&& _selected.find(item) == _selected.cend()) {
return; return;
} }
_selected.emplace(item, FullSelection); _selected.emplace(item, FullSelection);
@ -2059,10 +2102,16 @@ void HistoryInner::onUpdateSelected() {
HistoryTextState dragState; HistoryTextState dragState;
ClickHandlerHost *lnkhost = nullptr; ClickHandlerHost *lnkhost = nullptr;
bool selectingText = (item == _mouseActionItem && item == App::hoveredItem() && !_selected.empty() && _selected.cbegin()->second != FullSelection); auto selectingText = (item == _mouseActionItem)
&& (item == App::hoveredItem())
&& !_selected.empty()
&& (_selected.cbegin()->second != FullSelection);
if (point.y() < _historyPaddingTop) { if (point.y() < _historyPaddingTop) {
if (_botAbout && !_botAbout->info->text.isEmpty() && _botAbout->height > 0) { if (_botAbout && !_botAbout->info->text.isEmpty() && _botAbout->height > 0) {
dragState = _botAbout->info->text.getState(point - _botAbout->rect.topLeft() - QPoint(st::msgPadding.left(), st::msgPadding.top() + st::botDescSkip + st::msgNameFont->height), _botAbout->width); dragState = HistoryTextState(nullptr, _botAbout->info->text.getState(
point - _botAbout->rect.topLeft() - QPoint(st::msgPadding.left(), st::msgPadding.top() + st::botDescSkip + st::msgNameFont->height),
_botAbout->width));
_dragStateItem = App::histItemById(dragState.itemId);
lnkhost = _botAbout.get(); lnkhost = _botAbout.get();
} }
} else if (item) { } else if (item) {
@ -2077,7 +2126,7 @@ void HistoryInner::onUpdateSelected() {
auto dateHeight = st::msgServicePadding.bottom() + st::msgServiceFont->height + st::msgServicePadding.top(); auto dateHeight = st::msgServicePadding.bottom() + st::msgServiceFont->height + st::msgServicePadding.top();
auto scrollDateOpacity = _scrollDateOpacity.current(_scrollDateShown ? 1. : 0.); auto scrollDateOpacity = _scrollDateOpacity.current(_scrollDateShown ? 1. : 0.);
enumerateDates([this, &dragState, &lnkhost, &point, scrollDateOpacity, dateHeight/*, lastDate, showFloatingBefore*/](not_null<HistoryItem*> item, int itemtop, int dateTop) { enumerateDates([&](not_null<HistoryItem*> item, int itemtop, int dateTop) {
// stop enumeration if the date is above our point // stop enumeration if the date is above our point
if (dateTop + dateHeight <= point.y()) { if (dateTop + dateHeight <= point.y()) {
return false; return false;
@ -2116,7 +2165,10 @@ void HistoryInner::onUpdateSelected() {
} else { } else {
static_cast<DateClickHandler*>(_scrollDateLink.data())->setDate(item->date.date()); static_cast<DateClickHandler*>(_scrollDateLink.data())->setDate(item->date.date());
} }
dragState.link = _scrollDateLink; dragState = HistoryTextState(
nullptr,
_scrollDateLink);
_dragStateItem = App::histItemById(dragState.itemId);
lnkhost = item; lnkhost = item;
} }
} }
@ -2132,11 +2184,12 @@ void HistoryInner::onUpdateSelected() {
selectingText = false; selectingText = false;
} }
dragState = item->getState(m, request); dragState = item->getState(m, request);
_dragStateItem = App::histItemById(dragState.itemId);
lnkhost = item; lnkhost = item;
if (!dragState.link && m.x() >= st::historyPhotoLeft && m.x() < st::historyPhotoLeft + st::msgPhotoSize) { if (!dragState.link && m.x() >= st::historyPhotoLeft && m.x() < st::historyPhotoLeft + st::msgPhotoSize) {
if (auto msg = item->toHistoryMessage()) { if (auto msg = item->toHistoryMessage()) {
if (msg->hasFromPhoto()) { if (msg->hasFromPhoto()) {
enumerateUserpics([&dragState, &lnkhost, &point](not_null<HistoryMessage*> message, int userpicTop) -> bool { enumerateUserpics([&](not_null<HistoryMessage*> message, int userpicTop) -> bool {
// stop enumeration if the userpic is below our point // stop enumeration if the userpic is below our point
if (userpicTop > point.y()) { if (userpicTop > point.y()) {
return false; return false;
@ -2144,7 +2197,10 @@ void HistoryInner::onUpdateSelected() {
// stop enumeration if we've found a userpic under the cursor // stop enumeration if we've found a userpic under the cursor
if (point.y() >= userpicTop && point.y() < userpicTop + st::msgPhotoSize) { if (point.y() >= userpicTop && point.y() < userpicTop + st::msgPhotoSize) {
dragState.link = message->displayFrom()->openLink(); dragState = HistoryTextState(
nullptr,
message->displayFrom()->openLink());
_dragStateItem = App::histItemById(dragState.itemId);
lnkhost = message; lnkhost = message;
return false; return false;
} }
@ -2171,7 +2227,7 @@ void HistoryInner::onUpdateSelected() {
} else if (_mouseCursorState == HistoryInTextCursorState && (_selected.empty() || _selected.cbegin()->second != FullSelection)) { } else if (_mouseCursorState == HistoryInTextCursorState && (_selected.empty() || _selected.cbegin()->second != FullSelection)) {
cur = style::cur_text; cur = style::cur_text;
} else if (_mouseCursorState == HistoryInDateCursorState) { } else if (_mouseCursorState == HistoryInDateCursorState) {
// cur = style::cur_cross; //cur = style::cur_cross;
} }
} else if (item) { } else if (item) {
if (_mouseAction == MouseAction::Selecting) { if (_mouseAction == MouseAction::Selecting) {
@ -2235,7 +2291,9 @@ void HistoryInner::onUpdateSelected() {
if (ClickHandler::getPressed()) { if (ClickHandler::getPressed()) {
cur = style::cur_pointer; cur = style::cur_pointer;
} else if (_mouseAction == MouseAction::Selecting && !_selected.empty() && _selected.cbegin()->second != FullSelection) { } else if ((_mouseAction == MouseAction::Selecting)
&& !_selected.empty()
&& (_selected.cbegin()->second != FullSelection)) {
if (!_dragSelFrom || !_dragSelTo) { if (!_dragSelFrom || !_dragSelTo) {
cur = style::cur_text; cur = style::cur_text;
} }
@ -2243,7 +2301,7 @@ void HistoryInner::onUpdateSelected() {
} }
// Voice message seek support. // Voice message seek support.
if (auto pressedItem = App::pressedLinkItem()) { if (const auto pressedItem = _dragStateItem) {
if (!pressedItem->detached()) { if (!pressedItem->detached()) {
if (pressedItem->history() == _history || pressedItem->history() == _migrated) { if (pressedItem->history() == _history || pressedItem->history() == _migrated) {
auto adjustedPoint = mapPointToItem(point, pressedItem); auto adjustedPoint = mapPointToItem(point, pressedItem);
@ -2370,25 +2428,125 @@ void HistoryInner::applyDragSelection() {
applyDragSelection(&_selected); applyDragSelection(&_selected);
} }
void HistoryInner::addSelectionRange(SelectedItems *toItems, int32 fromblock, int32 fromitem, int32 toblock, int32 toitem, History *h) const { bool HistoryInner::isFullSelected(
not_null<SelectedItems*> toItems,
not_null<HistoryItem*> item) const {
const auto group = [&] {
if (const auto group = item->Get<HistoryMessageGroup>()) {
if (group->leader == item) {
return group;
}
return group->leader->Get<HistoryMessageGroup>();
}
return (HistoryMessageGroup*)nullptr;
}();
const auto singleSelected = [&](not_null<HistoryItem*> item) {
const auto i = toItems->find(item);
return (i != toItems->cend()) && (i->second == FullSelection);
};
if (group) {
if (!singleSelected(group->leader)) {
return false;
}
for (const auto other : group->others) {
if (!singleSelected(other)) {
return false;
}
}
} else if (!singleSelected(item)) {
return false;
}
return true;
}
void HistoryInner::changeDragSelection(
not_null<SelectedItems*> toItems,
not_null<HistoryItem*> item,
SelectAction action) const {
if (action == SelectAction::Invert) {
action = isFullSelected(toItems, item)
? SelectAction::Deselect
: SelectAction::Select;
}
auto total = toItems->size();
const auto add = (action == SelectAction::Select);
const auto goodForAdding = [&](not_null<HistoryItem*> item) {
if (item->id <= 0 || item->serviceMsg()) {
return false;
}
if (toItems->find(item) == toItems->end()) {
++total;
}
return true;
};
const auto addSingle = [&](not_null<HistoryItem*> item) {
const auto i = toItems->find(item);
if (i == toItems->cend()) {
toItems->emplace(item, FullSelection);
} else if (i->second != FullSelection) {
i->second = FullSelection;
}
};
const auto removeSingle = [&](not_null<HistoryItem*> item) {
const auto i = toItems->find(item);
if (i != toItems->cend()) {
toItems->erase(i);
}
};
const auto group = [&] {
if (const auto group = item->Get<HistoryMessageGroup>()) {
if (group->leader == item) {
return group;
}
return group->leader->Get<HistoryMessageGroup>();
}
return (HistoryMessageGroup*)nullptr;
}();
if (group) {
const auto adding = [&] {
if (!add || !goodForAdding(group->leader)) {
return false;
}
for (const auto other : group->others) {
if (!goodForAdding(other)) {
return false;
}
}
return (total <= MaxSelectedItems);
}();
if (adding) {
addSingle(group->leader);
for (const auto other : group->others) {
addSingle(other);
}
} else {
removeSingle(group->leader);
for (const auto other : group->others) {
removeSingle(other);
}
}
} else {
if (add && goodForAdding(item) && total <= MaxSelectedItems) {
addSingle(item);
} else {
removeSingle(item);
}
}
}
void HistoryInner::addSelectionRange(
not_null<SelectedItems*> toItems,
not_null<History*> history,
int fromblock,
int fromitem,
int toblock,
int toitem) const {
if (fromblock >= 0 && fromitem >= 0 && toblock >= 0 && toitem >= 0) { if (fromblock >= 0 && fromitem >= 0 && toblock >= 0 && toitem >= 0) {
for (; fromblock <= toblock; ++fromblock) { for (; fromblock <= toblock; ++fromblock) {
auto block = h->blocks[fromblock]; auto block = history->blocks[fromblock];
for (int32 cnt = (fromblock < toblock) ? block->items.size() : (toitem + 1); fromitem < cnt; ++fromitem) { for (int cnt = (fromblock < toblock) ? block->items.size() : (toitem + 1); fromitem < cnt; ++fromitem) {
auto item = block->items[fromitem]; auto item = block->items[fromitem];
auto i = toItems->find(item); changeDragSelection(toItems, item, SelectAction::Select);
if (item->id > 0 && !item->serviceMsg()) {
if (i == toItems->cend()) {
if (toItems->size() >= MaxSelectedItems) break;
toItems->emplace(item, FullSelection);
} else if (i->second != FullSelection) {
i->second = FullSelection;
}
} else {
if (i != toItems->cend()) {
toItems->erase(i);
}
}
} }
if (toItems->size() >= MaxSelectedItems) break; if (toItems->size() >= MaxSelectedItems) break;
fromitem = 0; fromitem = 0;
@ -2396,27 +2554,33 @@ void HistoryInner::addSelectionRange(SelectedItems *toItems, int32 fromblock, in
} }
} }
void HistoryInner::applyDragSelection(SelectedItems *toItems) const { void HistoryInner::applyDragSelection(
int32 selfromy = itemTop(_dragSelFrom), seltoy = itemTop(_dragSelTo); not_null<SelectedItems*> toItems) const {
const auto selfromy = itemTop(_dragSelFrom);
const auto seltoy = [&] {
auto result = itemTop(_dragSelTo);
return (result < 0) ? result : (result + _dragSelTo->height());
}();
if (selfromy < 0 || seltoy < 0) { if (selfromy < 0 || seltoy < 0) {
return; return;
} }
seltoy += _dragSelTo->height();
if (!toItems->empty() && toItems->cbegin()->second != FullSelection) { if (!toItems->empty() && toItems->cbegin()->second != FullSelection) {
toItems->clear(); toItems->clear();
} }
if (_dragSelecting) { if (_dragSelecting) {
int32 fromblock = _dragSelFrom->block()->indexInHistory(), fromitem = _dragSelFrom->indexInBlock(); auto fromblock = _dragSelFrom->block()->indexInHistory();
int32 toblock = _dragSelTo->block()->indexInHistory(), toitem = _dragSelTo->indexInBlock(); auto fromitem = _dragSelFrom->indexInBlock();
auto toblock = _dragSelTo->block()->indexInHistory();
auto toitem = _dragSelTo->indexInBlock();
if (_migrated) { if (_migrated) {
if (_dragSelFrom->history() == _migrated) { if (_dragSelFrom->history() == _migrated) {
if (_dragSelTo->history() == _migrated) { if (_dragSelTo->history() == _migrated) {
addSelectionRange(toItems, fromblock, fromitem, toblock, toitem, _migrated); addSelectionRange(toItems, _migrated, fromblock, fromitem, toblock, toitem);
toblock = -1; toblock = -1;
toitem = -1; toitem = -1;
} else { } else {
addSelectionRange(toItems, fromblock, fromitem, _migrated->blocks.size() - 1, _migrated->blocks.back()->items.size() - 1, _migrated); addSelectionRange(toItems, _migrated, fromblock, fromitem, _migrated->blocks.size() - 1, _migrated->blocks.back()->items.size() - 1);
} }
fromblock = 0; fromblock = 0;
fromitem = 0; fromitem = 0;
@ -2425,20 +2589,20 @@ void HistoryInner::applyDragSelection(SelectedItems *toItems) const {
toitem = -1; toitem = -1;
} }
} }
addSelectionRange(toItems, fromblock, fromitem, toblock, toitem, _history); addSelectionRange(toItems, _history, fromblock, fromitem, toblock, toitem);
} else { } else {
for (auto i = toItems->begin(); i != toItems->cend();) { auto toRemove = std::vector<not_null<HistoryItem*>>();
for (auto i = toItems->begin(); i != toItems->cend(); ++i) {
auto iy = itemTop(i->first); auto iy = itemTop(i->first);
if (iy < 0) { if (iy < -1) {
if (iy < -1) i = toItems->erase(i); toRemove.push_back(i->first);
continue; } else if (iy >= 0 && iy >= selfromy && iy < seltoy) {
} toRemove.push_back(i->first);
if (iy >= selfromy && iy < seltoy) {
i = toItems->erase(i);
} else {
++i;
} }
} }
for (const auto item : toRemove) {
changeDragSelection(toItems, item, SelectAction::Deselect);
}
} }
} }

View File

@ -66,7 +66,7 @@ public:
HistoryTopBarWidget::SelectedState getSelectionState() const; HistoryTopBarWidget::SelectedState getSelectionState() const;
void clearSelectedItems(bool onlyTextSelection = false); void clearSelectedItems(bool onlyTextSelection = false);
MessageIdsList getSelectedItems() const; MessageIdsList getSelectedItems() const;
void selectItem(HistoryItem *item); void selectItem(not_null<HistoryItem*> item);
void updateBotInfo(bool recount = true); void updateBotInfo(bool recount = true);
@ -165,6 +165,10 @@ private:
HistoryItem *prevItem(HistoryItem *item); HistoryItem *prevItem(HistoryItem *item);
HistoryItem *nextItem(HistoryItem *item); HistoryItem *nextItem(HistoryItem *item);
void updateDragSelection(HistoryItem *dragSelFrom, HistoryItem *dragSelTo, bool dragSelecting); void updateDragSelection(HistoryItem *dragSelFrom, HistoryItem *dragSelTo, bool dragSelecting);
TextSelection itemRenderSelection(
not_null<HistoryItem*> item,
int selfromy,
int seltoy) const;
void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard); void setToClipboard(const TextWithEntities &forClipboard, QClipboard::Mode mode = QClipboard::Clipboard);
@ -217,8 +221,26 @@ private:
using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>; using SelectedItems = std::map<HistoryItem*, TextSelection, std::less<>>;
SelectedItems _selected; SelectedItems _selected;
void applyDragSelection(); void applyDragSelection();
void applyDragSelection(SelectedItems *toItems) const; void applyDragSelection(not_null<SelectedItems*> toItems) const;
void addSelectionRange(SelectedItems *toItems, int32 fromblock, int32 fromitem, int32 toblock, int32 toitem, History *h) const; void addSelectionRange(
not_null<SelectedItems*> toItems,
not_null<History*> history,
int fromblock,
int fromitem,
int toblock,
int toitem) const;
bool isFullSelected(
not_null<SelectedItems*> toItems,
not_null<HistoryItem*> item) const;
enum class SelectAction {
Select,
Deselect,
Invert,
};
void changeDragSelection(
not_null<SelectedItems*> toItems,
not_null<HistoryItem*> item,
SelectAction action) const;
// Does any of the shown histories has this flag set. // Does any of the shown histories has this flag set.
bool hasPendingResizedItems() const { bool hasPendingResizedItems() const {
@ -230,6 +252,7 @@ private:
QPoint _dragStartPosition; QPoint _dragStartPosition;
QPoint _mousePosition; QPoint _mousePosition;
HistoryItem *_mouseActionItem = nullptr; HistoryItem *_mouseActionItem = nullptr;
HistoryItem *_dragStateItem = nullptr;
HistoryCursorState _mouseCursorState = HistoryDefaultCursorState; HistoryCursorState _mouseCursorState = HistoryDefaultCursorState;
uint16 _mouseTextSymbol = 0; uint16 _mouseTextSymbol = 0;
bool _pressWasInactive = false; bool _pressWasInactive = false;

View File

@ -46,6 +46,29 @@ constexpr int kAttachMessageToPreviousSecondsDelta = 900;
} // namespace } // namespace
HistoryTextState::HistoryTextState(not_null<const HistoryItem*> item)
: itemId(item->fullId()) {
}
HistoryTextState::HistoryTextState(
not_null<const HistoryItem*> item,
const Text::StateResult &state)
: itemId(item->fullId())
, cursor(state.uponSymbol
? HistoryInTextCursorState
: HistoryDefaultCursorState)
, link(state.link)
, afterSymbol(state.afterSymbol)
, symbol(state.symbol) {
}
HistoryTextState::HistoryTextState(
not_null<const HistoryItem*> item,
ClickHandlerPtr link)
: itemId(item->fullId())
, link(link) {
}
ReplyMarkupClickHandler::ReplyMarkupClickHandler(const HistoryItem *item, int row, int col) ReplyMarkupClickHandler::ReplyMarkupClickHandler(const HistoryItem *item, int row, int col)
: _itemId(item->fullId()) : _itemId(item->fullId())
, _row(row) , _row(row)

View File

@ -79,25 +79,33 @@ enum HistoryCursorState {
struct HistoryTextState { struct HistoryTextState {
HistoryTextState() = default; HistoryTextState() = default;
HistoryTextState(const Text::StateResult &state) HistoryTextState(not_null<const HistoryItem*> item);
: cursor(state.uponSymbol ? HistoryInTextCursorState : HistoryDefaultCursorState) HistoryTextState(
, link(state.link) not_null<const HistoryItem*> item,
, afterSymbol(state.afterSymbol) const Text::StateResult &state);
, symbol(state.symbol) { HistoryTextState(
not_null<const HistoryItem*> item,
ClickHandlerPtr link);
HistoryTextState(
std::nullptr_t,
const Text::StateResult &state)
: cursor(state.uponSymbol
? HistoryInTextCursorState
: HistoryDefaultCursorState)
, link(state.link)
, afterSymbol(state.afterSymbol)
, symbol(state.symbol) {
} }
HistoryTextState &operator=(const Text::StateResult &state) { HistoryTextState(std::nullptr_t, ClickHandlerPtr link)
cursor = state.uponSymbol ? HistoryInTextCursorState : HistoryDefaultCursorState; : link(link) {
link = state.link;
afterSymbol = state.afterSymbol;
symbol = state.symbol;
return *this;
}
HistoryTextState(ClickHandlerPtr link) : link(link) {
} }
FullMsgId itemId;
HistoryCursorState cursor = HistoryDefaultCursorState; HistoryCursorState cursor = HistoryDefaultCursorState;
ClickHandlerPtr link; ClickHandlerPtr link;
bool afterSymbol = false; bool afterSymbol = false;
uint16 symbol = 0; uint16 symbol = 0;
}; };
struct HistoryStateRequest { struct HistoryStateRequest {

View File

@ -135,7 +135,8 @@ public:
return false; return false;
} }
virtual std::unique_ptr<HistoryMedia> clone( virtual std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const = 0; not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const = 0;
virtual DocumentData *getDocument() { virtual DocumentData *getDocument() {
return nullptr; return nullptr;

View File

@ -54,13 +54,20 @@ HistoryGroupedMedia::Element::Element(not_null<HistoryItem*> item)
HistoryGroupedMedia::HistoryGroupedMedia( HistoryGroupedMedia::HistoryGroupedMedia(
not_null<HistoryItem*> parent, not_null<HistoryItem*> parent,
const std::vector<not_null<HistoryItem*>> &others) const std::vector<not_null<HistoryItem*>> &others)
: HistoryMedia(parent) { : HistoryMedia(parent)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
const auto result = applyGroup(others); const auto result = applyGroup(others);
Ensures(result); Ensures(result);
} }
void HistoryGroupedMedia::initDimensions() { void HistoryGroupedMedia::initDimensions() {
if (_caption.hasSkipBlock()) {
_caption.setSkipBlock(
_parent->skipBlockWidth(),
_parent->skipBlockHeight());
}
std::vector<QSize> sizes; std::vector<QSize> sizes;
sizes.reserve(_elements.size()); sizes.reserve(_elements.size());
for (const auto &element : _elements) { for (const auto &element : _elements) {
@ -84,6 +91,14 @@ void HistoryGroupedMedia::initDimensions() {
_elements[i].initialGeometry = item.geometry; _elements[i].initialGeometry = item.geometry;
_elements[i].sides = item.sides; _elements[i].sides = item.sides;
} }
if (!_caption.isEmpty()) {
auto captionw = _maxw - st::msgPadding.left() - st::msgPadding.right();
_minh += st::mediaCaptionSkip + _caption.countHeight(captionw);
if (isBubbleBottom()) {
_minh += st::msgPadding.bottom();
}
}
} }
int HistoryGroupedMedia::resizeGetHeight(int width) { int HistoryGroupedMedia::resizeGetHeight(int width) {
@ -94,7 +109,7 @@ int HistoryGroupedMedia::resizeGetHeight(int width) {
} }
const auto initialSpacing = st::historyGroupSkip; const auto initialSpacing = st::historyGroupSkip;
const auto factor = width / float64(st::historyGroupWidthMax); const auto factor = width / float64(_maxw);
const auto scale = [&](int value) { const auto scale = [&](int value) {
return int(std::round(value * factor)); return int(std::round(value * factor));
}; };
@ -124,6 +139,15 @@ int HistoryGroupedMedia::resizeGetHeight(int width) {
accumulate_max(_height, top + height); accumulate_max(_height, top + height);
} }
if (!_caption.isEmpty()) {
const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right();
_height += st::mediaPadding.bottom() + st::mediaCaptionSkip + _caption.countHeight(captionw);
if (isBubbleBottom()) {
_height += st::msgPadding.bottom();
}
}
return _height; return _height;
} }
@ -132,7 +156,13 @@ void HistoryGroupedMedia::draw(
const QRect &clip, const QRect &clip,
TextSelection selection, TextSelection selection,
TimeMs ms) const { TimeMs ms) const {
for (const auto &element : _elements) { for (auto i = 0, count = int(_elements.size()); i != count; ++i) {
const auto &element = _elements[i];
const auto elementSelection = (selection == FullSelection)
? FullSelection
: IsGroupItemSelection(selection, i)
? FullSelection
: TextSelection();
auto corners = GetCornersFromSides(element.sides); auto corners = GetCornersFromSides(element.sides);
if (!isBubbleTop()) { if (!isBubbleTop()) {
corners &= ~(RectPart::TopLeft | RectPart::TopRight); corners &= ~(RectPart::TopLeft | RectPart::TopRight);
@ -143,13 +173,36 @@ void HistoryGroupedMedia::draw(
element.content->drawGrouped( element.content->drawGrouped(
p, p,
clip, clip,
selection, elementSelection,
ms, ms,
element.geometry, element.geometry,
corners, corners,
&element.cacheKey, &element.cacheKey,
&element.cache); &element.cache);
} }
// date
const auto selected = (selection == FullSelection);
if (!_caption.isEmpty()) {
const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right();
const auto outbg = _parent->hasOutLayout();
const auto captiony = _height
- (isBubbleBottom() ? st::msgPadding.bottom() : 0)
- _caption.countHeight(captionw);
p.setPen(outbg ? (selected ? st::historyTextOutFgSelected : st::historyTextOutFg) : (selected ? st::historyTextInFgSelected : st::historyTextInFg));
_caption.draw(p, st::msgPadding.left(), captiony, captionw, style::al_left, 0, -1, selection);
} else if (_parent->getMedia() == this) {
auto fullRight = _width;
auto fullBottom = _height;
if (_parent->id < 0 || App::hoveredItem() == _parent) {
_parent->drawInfo(p, fullRight, fullBottom, _width, selected, InfoDisplayOverImage);
}
if (!_parent->hasBubble() && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
_parent->drawRightAction(p, fastShareLeft, fastShareTop, _width);
}
}
} }
HistoryTextState HistoryGroupedMedia::getState( HistoryTextState HistoryGroupedMedia::getState(
@ -157,10 +210,39 @@ HistoryTextState HistoryGroupedMedia::getState(
HistoryStateRequest request) const { HistoryStateRequest request) const {
for (const auto &element : _elements) { for (const auto &element : _elements) {
if (element.geometry.contains(point)) { if (element.geometry.contains(point)) {
return element.content->getStateGrouped( auto result = element.content->getStateGrouped(
element.geometry, element.geometry,
point, point,
request); request);
result.itemId = element.item->fullId();
return result;
}
}
if (!_caption.isEmpty()) {
const auto captionw = _width - st::msgPadding.left() - st::msgPadding.right();
const auto captiony = _height
- (isBubbleBottom() ? st::msgPadding.bottom() : 0)
- _caption.countHeight(captionw);
if (QRect(st::msgPadding.left(), captiony, captionw, _height - captiony).contains(point)) {
return HistoryTextState(_parent, _caption.getState(
point - QPoint(st::msgPadding.left(), captiony),
captionw,
request.forText()));
}
}
auto result = HistoryTextState(_parent);
if (_caption.isEmpty() && _parent->getMedia() == this) {
auto fullRight = _width;
auto fullBottom = _height;
if (_parent->pointInTime(fullRight, fullBottom, point, InfoDisplayOverImage)) {
result.cursor = HistoryInDateCursorState;
}
if (!_parent->hasBubble() && _parent->displayRightAction()) {
auto fastShareLeft = (fullRight + st::historyFastShareLeft);
auto fastShareTop = (fullBottom - st::historyFastShareBottom - st::historyFastShareSize);
if (QRect(fastShareLeft, fastShareTop, st::historyFastShareSize, st::historyFastShareSize).contains(point)) {
result.link = _parent->rightActionLink();
}
} }
} }
return HistoryTextState(); return HistoryTextState();
@ -191,6 +273,14 @@ TextSelection HistoryGroupedMedia::adjustSelection(
return _caption.adjustSelection(selection, type); return _caption.adjustSelection(selection, type);
} }
QString HistoryGroupedMedia::notificationText() const {
return WithCaptionNotificationText(lang(lng_in_dlg_album), _caption);
}
QString HistoryGroupedMedia::inDialogsText() const {
return WithCaptionDialogsText(lang(lng_in_dlg_album), _caption);
}
TextWithEntities HistoryGroupedMedia::selectedText( TextWithEntities HistoryGroupedMedia::selectedText(
TextSelection selection) const { TextSelection selection) const {
return WithCaptionSelectedText( return WithCaptionSelectedText(
@ -212,6 +302,9 @@ void HistoryGroupedMedia::clickHandlerPressedChanged(
bool pressed) { bool pressed) {
for (const auto &element : _elements) { for (const auto &element : _elements) {
element.content->clickHandlerPressedChanged(p, pressed); element.content->clickHandlerPressedChanged(p, pressed);
if (pressed && element.content->dragItemByHandler(p)) {
App::pressedLinkItem(element.item);
}
} }
} }
@ -243,7 +336,7 @@ bool HistoryGroupedMedia::applyGroup(
Assert(media != nullptr && media->canBeGrouped()); Assert(media != nullptr && media->canBeGrouped());
_elements.push_back(Element(item)); _elements.push_back(Element(item));
_elements.back().content = item->getMedia()->clone(_parent); _elements.back().content = item->getMedia()->clone(_parent, item);
}; };
if (_elements.empty()) { if (_elements.empty()) {
pushElement(_parent); pushElement(_parent);

View File

@ -34,7 +34,8 @@ public:
return MediaTypeGrouped; return MediaTypeGrouped;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Unexpected("Clone HistoryGroupedMedia."); Unexpected("Clone HistoryGroupedMedia.");
} }
@ -64,6 +65,8 @@ public:
return !_caption.isEmpty(); return !_caption.isEmpty();
} }
QString notificationText() const override;
QString inDialogsText() const override;
TextWithEntities selectedText(TextSelection selection) const override; TextWithEntities selectedText(TextSelection selection) const override;
void clickHandlerActiveChanged( void clickHandlerActiveChanged(

View File

@ -272,12 +272,16 @@ HistoryPhoto::HistoryPhoto(
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, _data(photo) , _data(photo)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) { , _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
const auto fullId = parent->fullId();
setLinks( setLinks(
MakeShared<PhotoOpenClickHandler>(_data), MakeShared<PhotoOpenClickHandler>(_data, fullId),
MakeShared<PhotoSaveClickHandler>(_data), MakeShared<PhotoSaveClickHandler>(_data, fullId),
MakeShared<PhotoCancelClickHandler>(_data)); MakeShared<PhotoCancelClickHandler>(_data, fullId));
if (!caption.isEmpty()) { if (!caption.isEmpty()) {
_caption.setText(st::messageTextStyle, caption + _parent->skipBlock(), itemTextNoMonoOptions(_parent)); _caption.setText(
st::messageTextStyle,
caption + _parent->skipBlock(),
itemTextNoMonoOptions(_parent));
} }
init(); init();
} }
@ -289,10 +293,11 @@ HistoryPhoto::HistoryPhoto(
int width) int width)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, _data(photo) { , _data(photo) {
const auto fullId = parent->fullId();
setLinks( setLinks(
MakeShared<PhotoOpenClickHandler>(_data, chat), MakeShared<PhotoOpenClickHandler>(_data, fullId, chat),
MakeShared<PhotoSaveClickHandler>(_data, chat), MakeShared<PhotoSaveClickHandler>(_data, fullId, chat),
MakeShared<PhotoCancelClickHandler>(_data, chat)); MakeShared<PhotoCancelClickHandler>(_data, fullId, chat));
_width = width; _width = width;
init(); init();
@ -308,16 +313,18 @@ HistoryPhoto::HistoryPhoto(
HistoryPhoto::HistoryPhoto( HistoryPhoto::HistoryPhoto(
not_null<HistoryItem*> parent, not_null<HistoryItem*> parent,
not_null<HistoryItem*> realParent,
const HistoryPhoto &other) const HistoryPhoto &other)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, _data(other._data) , _data(other._data)
, _pixw(other._pixw) , _pixw(other._pixw)
, _pixh(other._pixh) , _pixh(other._pixh)
, _caption(other._caption) { , _caption(other._caption) {
const auto fullId = realParent->fullId();
setLinks( setLinks(
MakeShared<PhotoOpenClickHandler>(_data), MakeShared<PhotoOpenClickHandler>(_data, fullId),
MakeShared<PhotoSaveClickHandler>(_data), MakeShared<PhotoSaveClickHandler>(_data, fullId),
MakeShared<PhotoCancelClickHandler>(_data)); MakeShared<PhotoCancelClickHandler>(_data, fullId));
init(); init();
} }
@ -533,9 +540,11 @@ void HistoryPhoto::draw(Painter &p, const QRect &r, TextSelection selection, Tim
} }
HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int skipx = 0, skipy = 0, width = _width, height = _height; int skipx = 0, skipy = 0, width = _width, height = _height;
auto bubble = _parent->hasBubble(); auto bubble = _parent->hasBubble();
@ -549,7 +558,10 @@ HistoryTextState HistoryPhoto::getState(QPoint point, HistoryStateRequest reques
height -= st::msgPadding.bottom(); height -= st::msgPadding.bottom();
} }
if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) {
result = _caption.getState(point - QPoint(st::msgPadding.left(), height), captionw, request.forText()); result = HistoryTextState(_parent, _caption.getState(
point - QPoint(st::msgPadding.left(), height),
captionw,
request.forText()));
return result; return result;
} }
height -= st::mediaCaptionSkip; height -= st::mediaCaptionSkip;
@ -696,7 +708,7 @@ HistoryTextState HistoryPhoto::getStateGrouped(
return {}; return {};
} }
const auto delayed = _data->full->toDelayedStorageImage(); const auto delayed = _data->full->toDelayedStorageImage();
return _data->uploading() return HistoryTextState(_parent, _data->uploading()
? _cancell ? _cancell
: _data->loaded() : _data->loaded()
? _openl ? _openl
@ -704,7 +716,7 @@ HistoryTextState HistoryPhoto::getStateGrouped(
? ((!delayed || !delayed->location().isNull()) ? ((!delayed || !delayed->location().isNull())
? _cancell ? _cancell
: ClickHandlerPtr()) : ClickHandlerPtr())
: _savel; : _savel);
} }
void HistoryPhoto::validateGroupedCache( void HistoryPhoto::validateGroupedCache(
@ -872,10 +884,13 @@ HistoryVideo::HistoryVideo(
, _thumbw(1) , _thumbw(1)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) { , _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
if (!caption.isEmpty()) { if (!caption.isEmpty()) {
_caption.setText(st::messageTextStyle, caption + _parent->skipBlock(), itemTextNoMonoOptions(_parent)); _caption.setText(
st::messageTextStyle,
caption + _parent->skipBlock(),
itemTextNoMonoOptions(_parent));
} }
setDocumentLinks(_data); setDocumentLinks(_data, parent);
setStatusSize(FileStatusSizeReady); setStatusSize(FileStatusSizeReady);
@ -884,12 +899,13 @@ HistoryVideo::HistoryVideo(
HistoryVideo::HistoryVideo( HistoryVideo::HistoryVideo(
not_null<HistoryItem*> parent, not_null<HistoryItem*> parent,
not_null<HistoryItem*> realParent,
const HistoryVideo &other) const HistoryVideo &other)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, _data(other._data) , _data(other._data)
, _thumbw(other._thumbw) , _thumbw(other._thumbw)
, _caption(other._caption) { , _caption(other._caption) {
setDocumentLinks(_data); setDocumentLinks(_data, realParent);
setStatusSize(other._statusSize); setStatusSize(other._statusSize);
} }
@ -1089,7 +1105,7 @@ HistoryTextState HistoryVideo::getState(QPoint point, HistoryStateRequest reques
return {}; return {};
} }
HistoryTextState result; auto result = HistoryTextState(_parent);
bool loaded = _data->loaded(); bool loaded = _data->loaded();
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
@ -1105,7 +1121,10 @@ HistoryTextState HistoryVideo::getState(QPoint point, HistoryStateRequest reques
height -= st::msgPadding.bottom(); height -= st::msgPadding.bottom();
} }
if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) {
result = _caption.getState(point - QPoint(st::msgPadding.left(), height), captionw, request.forText()); result = HistoryTextState(_parent, _caption.getState(
point - QPoint(st::msgPadding.left(), height),
captionw,
request.forText()));
} }
height -= st::mediaCaptionSkip; height -= st::mediaCaptionSkip;
} }
@ -1242,13 +1261,13 @@ HistoryTextState HistoryVideo::getStateGrouped(
if (!geometry.contains(point)) { if (!geometry.contains(point)) {
return {}; return {};
} }
return _data->uploading() return HistoryTextState(_parent, _data->uploading()
? _cancell ? _cancell
: _data->loaded() : _data->loaded()
? _openl ? _openl
: _data->loading() : _data->loading()
? _cancell ? _cancell
: _savel; : _savel);
} }
void HistoryVideo::validateGroupedCache( void HistoryVideo::validateGroupedCache(
@ -1425,7 +1444,7 @@ HistoryDocument::HistoryDocument(
fillNamedFromData(named); fillNamedFromData(named);
} }
setDocumentLinks(_data); setDocumentLinks(_data, parent);
setStatusSize(FileStatusSizeReady); setStatusSize(FileStatusSizeReady);
@ -1450,7 +1469,7 @@ HistoryDocument::HistoryDocument(
} }
} }
setDocumentLinks(_data); setDocumentLinks(_data, parent);
setStatusSize(other._statusSize); setStatusSize(other._statusSize);
@ -1821,9 +1840,11 @@ void HistoryDocument::draw(Painter &p, const QRect &r, TextSelection selection,
} }
HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
bool loaded = _data->loaded(); bool loaded = _data->loaded();
@ -1882,7 +1903,10 @@ HistoryTextState HistoryDocument::getState(QPoint point, HistoryStateRequest req
auto height = _height; auto height = _height;
if (auto captioned = Get<HistoryDocumentCaptioned>()) { if (auto captioned = Get<HistoryDocumentCaptioned>()) {
if (point.y() >= bottom) { if (point.y() >= bottom) {
result = captioned->_caption.getState(point - QPoint(st::msgPadding.left(), bottom), _width - st::msgPadding.left() - st::msgPadding.right(), request.forText()); result = HistoryTextState(_parent, captioned->_caption.getState(
point - QPoint(st::msgPadding.left(), bottom),
_width - st::msgPadding.left() - st::msgPadding.right(),
request.forText()));
return result; return result;
} }
auto captionw = _width - st::msgPadding.left() - st::msgPadding.right(); auto captionw = _width - st::msgPadding.left() - st::msgPadding.right();
@ -2163,7 +2187,7 @@ HistoryGif::HistoryGif(
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, _data(document) , _data(document)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) { , _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
setDocumentLinks(_data, true); setDocumentLinks(_data, parent, true);
setStatusSize(FileStatusSizeReady); setStatusSize(FileStatusSizeReady);
@ -2182,7 +2206,7 @@ HistoryGif::HistoryGif(
, _thumbw(other._thumbw) , _thumbw(other._thumbw)
, _thumbh(other._thumbh) , _thumbh(other._thumbh)
, _caption(other._caption) { , _caption(other._caption) {
setDocumentLinks(_data, true); setDocumentLinks(_data, parent, true);
setStatusSize(other._statusSize); setStatusSize(other._statusSize);
} }
@ -2614,9 +2638,11 @@ void HistoryGif::draw(Painter &p, const QRect &r, TextSelection selection, TimeM
} }
HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
@ -2630,7 +2656,10 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
height -= st::msgPadding.bottom(); height -= st::msgPadding.bottom();
} }
if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) { if (QRect(st::msgPadding.left(), height, captionw, _height - height).contains(point)) {
result = _caption.getState(point - QPoint(st::msgPadding.left(), height), captionw, request.forText()); result = HistoryTextState(_parent, _caption.getState(
point - QPoint(st::msgPadding.left(), height),
captionw,
request.forText()));
return result; return result;
} }
height -= st::mediaCaptionSkip; height -= st::mediaCaptionSkip;
@ -2679,7 +2708,10 @@ HistoryTextState HistoryGif::getState(QPoint point, HistoryStateRequest request)
if (breakEverywhere) { if (breakEverywhere) {
textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere; textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere;
} }
result = forwarded->_text.getState(point - QPoint(rectx + st::msgReplyPadding.left(), recty + st::msgReplyPadding.top()), innerw, textRequest); result = HistoryTextState(_parent, forwarded->_text.getState(
point - QPoint(rectx + st::msgReplyPadding.left(), recty + st::msgReplyPadding.top()),
innerw,
textRequest));
result.symbol = 0; result.symbol = 0;
result.afterSymbol = false; result.afterSymbol = false;
if (breakEverywhere) { if (breakEverywhere) {
@ -3138,8 +3170,10 @@ void HistorySticker::draw(Painter &p, const QRect &r, TextSelection selection, T
} }
HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistorySticker::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
auto outbg = _parent->hasOutLayout(); auto outbg = _parent->hasOutLayout();
auto childmedia = (_parent->getMedia() != this); auto childmedia = (_parent->getMedia() != this);
@ -3408,7 +3442,7 @@ void HistoryContact::draw(Painter &p, const QRect &r, TextSelection selection, T
} }
HistoryTextState HistoryContact::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryContact::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0; int32 nameleft = 0, nametop = 0, nameright = 0, statustop = 0, linktop = 0;
auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus; auto topMinus = isBubbleTop() ? 0 : st::msgFileTopMinus;
@ -3556,7 +3590,7 @@ void HistoryCall::draw(Painter &p, const QRect &r, TextSelection selection, Time
} }
HistoryTextState HistoryCall::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryCall::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (QRect(0, 0, _width, _height).contains(point)) { if (QRect(0, 0, _width, _height).contains(point)) {
result.link = _link; result.link = _link;
return result; return result;
@ -3617,9 +3651,12 @@ HistoryWebPage::HistoryWebPage(not_null<HistoryItem*> parent, not_null<WebPageDa
, _description(st::msgMinWidth - st::webPageLeft) { , _description(st::msgMinWidth - st::webPageLeft) {
} }
HistoryWebPage::HistoryWebPage(not_null<HistoryItem*> parent, const HistoryWebPage &other) : HistoryMedia(parent) HistoryWebPage::HistoryWebPage(
not_null<HistoryItem*> parent,
const HistoryWebPage &other)
: HistoryMedia(parent)
, _data(other._data) , _data(other._data)
, _attach(other._attach ? other._attach->clone(parent) : nullptr) , _attach(other._attach ? other._attach->clone(parent, parent) : nullptr)
, _asArticle(other._asArticle) , _asArticle(other._asArticle)
, _title(other._title) , _title(other._title)
, _description(other._description) , _description(other._description)
@ -3999,9 +4036,11 @@ void HistoryWebPage::draw(Painter &p, const QRect &r, TextSelection selection, T
} }
HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins());
@ -4030,7 +4069,11 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ
if (point.y() >= tshift && point.y() < tshift + _titleLines * lineHeight) { if (point.y() >= tshift && point.y() < tshift + _titleLines * lineHeight) {
Text::StateRequestElided titleRequest = request.forText(); Text::StateRequestElided titleRequest = request.forText();
titleRequest.lines = _titleLines; titleRequest.lines = _titleLines;
result = _title.getStateElidedLeft(point - QPoint(padding.left(), tshift), width, _width, titleRequest); result = HistoryTextState(_parent, _title.getStateElidedLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
titleRequest));
} else if (point.y() >= tshift + _titleLines * lineHeight) { } else if (point.y() >= tshift + _titleLines * lineHeight) {
symbolAdd += _title.length(); symbolAdd += _title.length();
} }
@ -4042,9 +4085,17 @@ HistoryTextState HistoryWebPage::getState(QPoint point, HistoryStateRequest requ
if (_descriptionLines > 0) { if (_descriptionLines > 0) {
Text::StateRequestElided descriptionRequest = request.forText(); Text::StateRequestElided descriptionRequest = request.forText();
descriptionRequest.lines = _descriptionLines; descriptionRequest.lines = _descriptionLines;
result = _description.getStateElidedLeft(point - QPoint(padding.left(), tshift), width, _width, descriptionRequest); result = HistoryTextState(_parent, _description.getStateElidedLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
descriptionRequest));
} else { } else {
result = _description.getStateLeft(point - QPoint(padding.left(), tshift), width, _width, request.forText()); result = HistoryTextState(_parent, _description.getStateLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
request.forText()));
} }
} else if (point.y() >= tshift + descriptionHeight) { } else if (point.y() >= tshift + descriptionHeight) {
symbolAdd += _description.length(); symbolAdd += _description.length();
@ -4178,7 +4229,7 @@ HistoryGame::HistoryGame(
const HistoryGame &other) const HistoryGame &other)
: HistoryMedia(parent) : HistoryMedia(parent)
, _data(other._data) , _data(other._data)
, _attach(other._attach ? other._attach->clone(parent) : nullptr) , _attach(other._attach ? other._attach->clone(parent, parent) : nullptr)
, _title(other._title) , _title(other._title)
, _description(other._description) { , _description(other._description) {
} }
@ -4395,9 +4446,11 @@ void HistoryGame::draw(Painter &p, const QRect &r, TextSelection selection, Time
} }
HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int32 width = _width, height = _height; int32 width = _width, height = _height;
QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins());
@ -4416,7 +4469,11 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request
if (point.y() >= tshift && point.y() < tshift + _titleLines * lineHeight) { if (point.y() >= tshift && point.y() < tshift + _titleLines * lineHeight) {
Text::StateRequestElided titleRequest = request.forText(); Text::StateRequestElided titleRequest = request.forText();
titleRequest.lines = _titleLines; titleRequest.lines = _titleLines;
result = _title.getStateElidedLeft(point - QPoint(padding.left(), tshift), width, _width, titleRequest); result = HistoryTextState(_parent, _title.getStateElidedLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
titleRequest));
} else if (point.y() >= tshift + _titleLines * lineHeight) { } else if (point.y() >= tshift + _titleLines * lineHeight) {
symbolAdd += _title.length(); symbolAdd += _title.length();
} }
@ -4426,7 +4483,11 @@ HistoryTextState HistoryGame::getState(QPoint point, HistoryStateRequest request
if (point.y() >= tshift && point.y() < tshift + _descriptionLines * lineHeight) { if (point.y() >= tshift && point.y() < tshift + _descriptionLines * lineHeight) {
Text::StateRequestElided descriptionRequest = request.forText(); Text::StateRequestElided descriptionRequest = request.forText();
descriptionRequest.lines = _descriptionLines; descriptionRequest.lines = _descriptionLines;
result = _description.getStateElidedLeft(point - QPoint(padding.left(), tshift), width, _width, descriptionRequest); result = HistoryTextState(_parent, _description.getStateElidedLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
descriptionRequest));
} else if (point.y() >= tshift + _descriptionLines * lineHeight) { } else if (point.y() >= tshift + _descriptionLines * lineHeight) {
symbolAdd += _description.length(); symbolAdd += _description.length();
} }
@ -4576,7 +4637,7 @@ HistoryInvoice::HistoryInvoice(
not_null<HistoryItem*> parent, not_null<HistoryItem*> parent,
const HistoryInvoice &other) const HistoryInvoice &other)
: HistoryMedia(parent) : HistoryMedia(parent)
, _attach(other._attach ? other._attach->clone(parent) : nullptr) , _attach(other._attach ? other._attach->clone(parent, parent) : nullptr)
, _titleHeight(other._titleHeight) , _titleHeight(other._titleHeight)
, _descriptionHeight(other._descriptionHeight) , _descriptionHeight(other._descriptionHeight)
, _title(other._title) , _title(other._title)
@ -4833,9 +4894,11 @@ void HistoryInvoice::draw(Painter &p, const QRect &r, TextSelection selection, T
} }
HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int32 width = _width, height = _height; int32 width = _width, height = _height;
QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins()); QMargins bubble(_attach ? _attach->bubbleMargins() : QMargins());
@ -4853,7 +4916,11 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ
if (point.y() >= tshift && point.y() < tshift + _titleHeight) { if (point.y() >= tshift && point.y() < tshift + _titleHeight) {
Text::StateRequestElided titleRequest = request.forText(); Text::StateRequestElided titleRequest = request.forText();
titleRequest.lines = _titleHeight / lineHeight; titleRequest.lines = _titleHeight / lineHeight;
result = _title.getStateElidedLeft(point - QPoint(padding.left(), tshift), width, _width, titleRequest); result = HistoryTextState(_parent, _title.getStateElidedLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
titleRequest));
} else if (point.y() >= tshift + _titleHeight) { } else if (point.y() >= tshift + _titleHeight) {
symbolAdd += _title.length(); symbolAdd += _title.length();
} }
@ -4861,7 +4928,11 @@ HistoryTextState HistoryInvoice::getState(QPoint point, HistoryStateRequest requ
} }
if (_descriptionHeight) { if (_descriptionHeight) {
if (point.y() >= tshift && point.y() < tshift + _descriptionHeight) { if (point.y() >= tshift && point.y() < tshift + _descriptionHeight) {
result = _description.getStateLeft(point - QPoint(padding.left(), tshift), width, _width, request.forText()); result = HistoryTextState(_parent, _description.getStateLeft(
point - QPoint(padding.left(), tshift),
width,
_width,
request.forText()));
} else if (point.y() >= tshift + _descriptionHeight) { } else if (point.y() >= tshift + _descriptionHeight) {
symbolAdd += _description.length(); symbolAdd += _description.length();
} }
@ -5122,10 +5193,12 @@ void HistoryLocation::draw(Painter &p, const QRect &r, TextSelection selection,
} }
HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(_parent);
auto symbolAdd = 0; auto symbolAdd = 0;
if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) return result; if (_width < st::msgPadding.left() + st::msgPadding.right() + 1) {
return result;
}
int32 skipx = 0, skipy = 0, width = _width, height = _height; int32 skipx = 0, skipy = 0, width = _width, height = _height;
bool bubble = _parent->hasBubble(); bool bubble = _parent->hasBubble();
@ -5145,7 +5218,11 @@ HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest req
if (!_title.isEmpty()) { if (!_title.isEmpty()) {
auto titleh = qMin(_title.countHeight(textw), 2 * st::webPageTitleFont->height); auto titleh = qMin(_title.countHeight(textw), 2 * st::webPageTitleFont->height);
if (point.y() >= skipy && point.y() < skipy + titleh) { if (point.y() >= skipy && point.y() < skipy + titleh) {
result = _title.getStateLeft(point - QPoint(skipx + st::msgPadding.left(), skipy), textw, _width, request.forText()); result = HistoryTextState(_parent, _title.getStateLeft(
point - QPoint(skipx + st::msgPadding.left(), skipy),
textw,
_width,
request.forText()));
return result; return result;
} else if (point.y() >= skipy + titleh) { } else if (point.y() >= skipy + titleh) {
symbolAdd += _title.length(); symbolAdd += _title.length();
@ -5155,7 +5232,11 @@ HistoryTextState HistoryLocation::getState(QPoint point, HistoryStateRequest req
if (!_description.isEmpty()) { if (!_description.isEmpty()) {
auto descriptionh = qMin(_description.countHeight(textw), 3 * st::webPageDescriptionFont->height); auto descriptionh = qMin(_description.countHeight(textw), 3 * st::webPageDescriptionFont->height);
if (point.y() >= skipy && point.y() < skipy + descriptionh) { if (point.y() >= skipy && point.y() < skipy + descriptionh) {
result = _description.getStateLeft(point - QPoint(skipx + st::msgPadding.left(), skipy), textw, _width, request.forText()); result = HistoryTextState(_parent, _description.getStateLeft(
point - QPoint(skipx + st::msgPadding.left(), skipy),
textw,
_width,
request.forText()));
} else if (point.y() >= skipy + descriptionh) { } else if (point.y() >= skipy + descriptionh) {
symbolAdd += _description.length(); symbolAdd += _description.length();
} }

View File

@ -72,21 +72,28 @@ public:
protected: protected:
ClickHandlerPtr _openl, _savel, _cancell; ClickHandlerPtr _openl, _savel, _cancell;
void setLinks(ClickHandlerPtr &&openl, ClickHandlerPtr &&savel, ClickHandlerPtr &&cancell); void setLinks(ClickHandlerPtr &&openl, ClickHandlerPtr &&savel, ClickHandlerPtr &&cancell);
void setDocumentLinks(DocumentData *document, bool inlinegif = false) { void setDocumentLinks(
not_null<DocumentData*> document,
not_null<HistoryItem*> realParent,
bool inlinegif = false) {
ClickHandlerPtr open, save; ClickHandlerPtr open, save;
const auto context = realParent->fullId();
if (inlinegif) { if (inlinegif) {
open = MakeShared<GifOpenClickHandler>(document); open = MakeShared<GifOpenClickHandler>(document, context);
} else { } else {
open = MakeShared<DocumentOpenClickHandler>(document); open = MakeShared<DocumentOpenClickHandler>(document, context);
} }
if (inlinegif) { if (inlinegif) {
save = MakeShared<GifOpenClickHandler>(document); save = MakeShared<GifOpenClickHandler>(document, context);
} else if (document->isVoiceMessage()) { } else if (document->isVoiceMessage()) {
save = MakeShared<DocumentOpenClickHandler>(document); save = MakeShared<DocumentOpenClickHandler>(document, context);
} else { } else {
save = MakeShared<DocumentSaveClickHandler>(document); save = MakeShared<DocumentSaveClickHandler>(document, context);
} }
setLinks(std::move(open), std::move(save), MakeShared<DocumentCancelClickHandler>(document)); setLinks(
std::move(open),
std::move(save),
MakeShared<DocumentCancelClickHandler>(document, context));
} }
// >= 0 will contain download / upload string, _statusSize = loaded bytes // >= 0 will contain download / upload string, _statusSize = loaded bytes
@ -153,15 +160,19 @@ public:
not_null<PeerData*> chat, not_null<PeerData*> chat,
const MTPDphoto &photo, const MTPDphoto &photo,
int width); int width);
HistoryPhoto(not_null<HistoryItem*> parent, const HistoryPhoto &other); HistoryPhoto(
not_null<HistoryItem*> parent,
not_null<HistoryItem*> realParent,
const HistoryPhoto &other);
void init(); void init();
HistoryMediaType type() const override { HistoryMediaType type() const override {
return MediaTypePhoto; return MediaTypePhoto;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
return std::make_unique<HistoryPhoto>(newParent, *this); not_null<HistoryItem*> realParent) const override {
return std::make_unique<HistoryPhoto>(newParent, realParent, *this);
} }
void initDimensions() override; void initDimensions() override;
@ -269,14 +280,18 @@ public:
not_null<HistoryItem*> parent, not_null<HistoryItem*> parent,
not_null<DocumentData*> document, not_null<DocumentData*> document,
const QString &caption); const QString &caption);
HistoryVideo(not_null<HistoryItem*> parent, const HistoryVideo &other); HistoryVideo(
not_null<HistoryItem*> parent,
not_null<HistoryItem*> realParent,
const HistoryVideo &other);
HistoryMediaType type() const override { HistoryMediaType type() const override {
return MediaTypeVideo; return MediaTypeVideo;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
return std::make_unique<HistoryVideo>(newParent, *this); not_null<HistoryItem*> realParent) const override {
return std::make_unique<HistoryVideo>(newParent, realParent, *this);
} }
void initDimensions() override; void initDimensions() override;
@ -460,7 +475,10 @@ public:
: MediaTypeFile); : MediaTypeFile);
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryDocument>(newParent, *this); return std::make_unique<HistoryDocument>(newParent, *this);
} }
@ -579,7 +597,10 @@ public:
return MediaTypeGif; return MediaTypeGif;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryGif>(newParent, *this); return std::make_unique<HistoryGif>(newParent, *this);
} }
@ -696,7 +717,10 @@ public:
return MediaTypeSticker; return MediaTypeSticker;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistorySticker>(newParent, _data); return std::make_unique<HistorySticker>(newParent, _data);
} }
@ -772,8 +796,16 @@ public:
return MediaTypeContact; return MediaTypeContact;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
return std::make_unique<HistoryContact>(newParent, _userId, _fname, _lname, _phone); not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryContact>(
newParent,
_userId,
_fname,
_lname,
_phone);
} }
void initDimensions() override; void initDimensions() override;
@ -840,7 +872,8 @@ public:
return MediaTypeCall; return MediaTypeCall;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Unexpected("Clone HistoryCall."); Unexpected("Clone HistoryCall.");
} }
@ -902,7 +935,10 @@ public:
return MediaTypeWebPage; return MediaTypeWebPage;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryWebPage>(newParent, *this); return std::make_unique<HistoryWebPage>(newParent, *this);
} }
@ -1012,7 +1048,10 @@ public:
return MediaTypeGame; return MediaTypeGame;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryGame>(newParent, *this); return std::make_unique<HistoryGame>(newParent, *this);
} }
@ -1127,7 +1166,10 @@ public:
return MediaTypeInvoice; return MediaTypeInvoice;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryInvoice>(newParent, *this); return std::make_unique<HistoryInvoice>(newParent, *this);
} }
@ -1231,7 +1273,10 @@ public:
return MediaTypeLocation; return MediaTypeLocation;
} }
std::unique_ptr<HistoryMedia> clone( std::unique_ptr<HistoryMedia> clone(
not_null<HistoryItem*> newParent) const override { not_null<HistoryItem*> newParent,
not_null<HistoryItem*> realParent) const override {
Expects(newParent == realParent);
return std::make_unique<HistoryLocation>(newParent, *this); return std::make_unique<HistoryLocation>(newParent, *this);
} }

View File

@ -757,7 +757,7 @@ HistoryMessage::HistoryMessage(
return (mediaType != MediaTypeCount); return (mediaType != MediaTypeCount);
}; };
if (cloneMedia()) { if (cloneMedia()) {
_media = mediaOriginal->clone(this); _media = mediaOriginal->clone(this, this);
} }
setText(fwd->originalText()); setText(fwd->originalText());
} }
@ -1512,12 +1512,22 @@ Storage::SharedMediaTypesMask HistoryMessage::sharedMediaTypes() const {
TextWithEntities HistoryMessage::selectedText(TextSelection selection) const { TextWithEntities HistoryMessage::selectedText(TextSelection selection) const {
TextWithEntities logEntryOriginalResult; TextWithEntities logEntryOriginalResult;
auto textResult = _text.originalTextWithEntities((selection == FullSelection) ? AllTextSelection : selection, ExpandLinksAll); const auto textSelection = (selection == FullSelection)
? AllTextSelection
: IsSubGroupSelection(selection)
? TextSelection(0, 0)
: selection;
auto textResult = _text.originalTextWithEntities(
textSelection,
ExpandLinksAll);
auto skipped = skipTextSelection(selection); auto skipped = skipTextSelection(selection);
auto mediaDisplayed = (_media && _media->isDisplayed()); auto mediaDisplayed = (_media && _media->isDisplayed());
auto mediaResult = mediaDisplayed ? _media->selectedText(skipped) : TextWithEntities(); auto mediaResult = mediaDisplayed ? _media->selectedText(skipped) : TextWithEntities();
if (auto entry = Get<HistoryMessageLogEntryOriginal>()) { if (auto entry = Get<HistoryMessageLogEntryOriginal>()) {
logEntryOriginalResult = entry->_page->selectedText(mediaDisplayed ? _media->skipSelection(skipped) : skipped); const auto originalSelection = mediaDisplayed
? _media->skipSelection(skipped)
: skipped;
logEntryOriginalResult = entry->_page->selectedText(originalSelection);
} }
auto result = textResult; auto result = textResult;
if (result.text.isEmpty()) { if (result.text.isEmpty()) {
@ -2232,7 +2242,7 @@ bool HistoryMessage::pointInTime(int right, int bottom, QPoint point, InfoDispla
} }
HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(this);
auto g = countGeometry(); auto g = countGeometry();
if (g.width() < 1) { if (g.width() < 1) {
@ -2272,7 +2282,9 @@ HistoryTextState HistoryMessage::getState(QPoint point, HistoryStateRequest requ
auto entryLeft = g.left(); auto entryLeft = g.left();
auto entryTop = trect.y() + trect.height(); auto entryTop = trect.y() + trect.height();
if (point.y() >= entryTop && point.y() < entryTop + entryHeight) { if (point.y() >= entryTop && point.y() < entryTop + entryHeight) {
result = entry->_page->getState(point - QPoint(entryLeft, entryTop), request); result = entry->_page->getState(
point - QPoint(entryLeft, entryTop),
request);
result.symbol += _text.length() + (mediaDisplayed ? _media->fullSelectionLength() : 0); result.symbol += _text.length() + (mediaDisplayed ? _media->fullSelectionLength() : 0);
} }
} }
@ -2408,7 +2420,10 @@ void HistoryMessage::updatePressed(QPoint point) {
} }
} }
bool HistoryMessage::getStateFromName(QPoint point, QRect &trect, HistoryTextState *outResult) const { bool HistoryMessage::getStateFromName(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult) const {
if (displayFromName()) { if (displayFromName()) {
if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) { if (point.y() >= trect.top() && point.y() < trect.top() + st::msgNameFont->height) {
auto user = displayFrom(); auto user = displayFrom();
@ -2428,7 +2443,11 @@ bool HistoryMessage::getStateFromName(QPoint point, QRect &trect, HistoryTextSta
return false; return false;
} }
bool HistoryMessage::getStateForwardedInfo(QPoint point, QRect &trect, HistoryTextState *outResult, const HistoryStateRequest &request) const { bool HistoryMessage::getStateForwardedInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult,
const HistoryStateRequest &request) const {
if (displayForwardedFrom()) { if (displayForwardedFrom()) {
auto forwarded = Get<HistoryMessageForwarded>(); auto forwarded = Get<HistoryMessageForwarded>();
auto fwdheight = ((forwarded->_text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height; auto fwdheight = ((forwarded->_text.maxWidth() > trect.width()) ? 2 : 1) * st::semiboldFont->height;
@ -2438,7 +2457,10 @@ bool HistoryMessage::getStateForwardedInfo(QPoint point, QRect &trect, HistoryTe
if (breakEverywhere) { if (breakEverywhere) {
textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere; textRequest.flags |= Text::StateRequest::Flag::BreakEverywhere;
} }
*outResult = forwarded->_text.getState(point - trect.topLeft(), trect.width(), textRequest); *outResult = HistoryTextState(this, forwarded->_text.getState(
point - trect.topLeft(),
trect.width(),
textRequest));
outResult->symbol = 0; outResult->symbol = 0;
outResult->afterSymbol = false; outResult->afterSymbol = false;
if (breakEverywhere) { if (breakEverywhere) {
@ -2453,7 +2475,10 @@ bool HistoryMessage::getStateForwardedInfo(QPoint point, QRect &trect, HistoryTe
return false; return false;
} }
bool HistoryMessage::getStateReplyInfo(QPoint point, QRect &trect, HistoryTextState *outResult) const { bool HistoryMessage::getStateReplyInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult) const {
if (auto reply = Get<HistoryMessageReply>()) { if (auto reply = Get<HistoryMessageReply>()) {
int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom(); int32 h = st::msgReplyPadding.top() + st::msgReplyBarSize.height() + st::msgReplyPadding.bottom();
if (point.y() >= trect.top() && point.y() < trect.top() + h) { if (point.y() >= trect.top() && point.y() < trect.top() + h) {
@ -2467,7 +2492,10 @@ bool HistoryMessage::getStateReplyInfo(QPoint point, QRect &trect, HistoryTextSt
return false; return false;
} }
bool HistoryMessage::getStateViaBotIdInfo(QPoint point, QRect &trect, HistoryTextState *outResult) const { bool HistoryMessage::getStateViaBotIdInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult) const {
if (!displayFromName() && !Has<HistoryMessageForwarded>()) { if (!displayFromName() && !Has<HistoryMessageForwarded>()) {
if (auto via = Get<HistoryMessageVia>()) { if (auto via = Get<HistoryMessageVia>()) {
if (QRect(trect.x(), trect.y(), via->_width, st::msgNameFont->height).contains(point)) { if (QRect(trect.x(), trect.y(), via->_width, st::msgNameFont->height).contains(point)) {
@ -2480,9 +2508,16 @@ bool HistoryMessage::getStateViaBotIdInfo(QPoint point, QRect &trect, HistoryTex
return false; return false;
} }
bool HistoryMessage::getStateText(QPoint point, QRect &trect, HistoryTextState *outResult, const HistoryStateRequest &request) const { bool HistoryMessage::getStateText(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult,
const HistoryStateRequest &request) const {
if (trect.contains(point)) { if (trect.contains(point)) {
*outResult = _text.getState(point - trect.topLeft(), trect.width(), request.forText()); *outResult = HistoryTextState(this, _text.getState(
point - trect.topLeft(),
trect.width(),
request.forText()));
return true; return true;
} }
return false; return false;
@ -2524,13 +2559,18 @@ TextSelection HistoryMessage::adjustSelection(TextSelection selection, TextSelec
} }
void HistoryMessage::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) { void HistoryMessage::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) {
if (_media) _media->clickHandlerActiveChanged(p, active);
HistoryItem::clickHandlerActiveChanged(p, active); HistoryItem::clickHandlerActiveChanged(p, active);
if (_media) {
_media->clickHandlerActiveChanged(p, active);
}
} }
void HistoryMessage::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) { void HistoryMessage::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
if (_media) _media->clickHandlerPressedChanged(p, pressed);
HistoryItem::clickHandlerPressedChanged(p, pressed); HistoryItem::clickHandlerPressedChanged(p, pressed);
if (_media) {
// HistoryGroupedMedia overrides HistoryItem App::pressedLinkItem().
_media->clickHandlerPressedChanged(p, pressed);
}
} }
QString HistoryMessage::notificationHeader() const { QString HistoryMessage::notificationHeader() const {

View File

@ -329,11 +329,28 @@ private:
void paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const; void paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const;
void paintText(Painter &p, QRect &trect, TextSelection selection) const; void paintText(Painter &p, QRect &trect, TextSelection selection) const;
bool getStateFromName(QPoint point, QRect &trect, HistoryTextState *outResult) const; bool getStateFromName(
bool getStateForwardedInfo(QPoint point, QRect &trect, HistoryTextState *outResult, const HistoryStateRequest &request) const; QPoint point,
bool getStateReplyInfo(QPoint point, QRect &trect, HistoryTextState *outResult) const; QRect &trect,
bool getStateViaBotIdInfo(QPoint point, QRect &trect, HistoryTextState *outResult) const; not_null<HistoryTextState*> outResult) const;
bool getStateText(QPoint point, QRect &trect, HistoryTextState *outResult, const HistoryStateRequest &request) const; bool getStateForwardedInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult,
const HistoryStateRequest &request) const;
bool getStateReplyInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult) const;
bool getStateViaBotIdInfo(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult) const;
bool getStateText(
QPoint point,
QRect &trect,
not_null<HistoryTextState*> outResult,
const HistoryStateRequest &request) const;
void setMedia(const MTPMessageMedia *media); void setMedia(const MTPMessageMedia *media);
void setReplyMarkup(const MTPReplyMarkup *markup); void setReplyMarkup(const MTPReplyMarkup *markup);

View File

@ -594,7 +594,7 @@ bool HistoryService::hasPoint(QPoint point) const {
} }
HistoryTextState HistoryService::getState(QPoint point, HistoryStateRequest request) const { HistoryTextState HistoryService::getState(QPoint point, HistoryStateRequest request) const {
HistoryTextState result; auto result = HistoryTextState(this);
auto g = countGeometry(); auto g = countGeometry();
if (g.width() < 1) { if (g.width() < 1) {
@ -618,7 +618,10 @@ HistoryTextState HistoryService::getState(QPoint point, HistoryStateRequest requ
if (trect.contains(point)) { if (trect.contains(point)) {
auto textRequest = request.forText(); auto textRequest = request.forText();
textRequest.align = style::al_center; textRequest.align = style::al_center;
result = _text.getState(point - trect.topLeft(), trect.width(), textRequest); result = HistoryTextState(this, _text.getState(
point - trect.topLeft(),
trect.width(),
textRequest));
if (auto gamescore = Get<HistoryServiceGameScore>()) { if (auto gamescore = Get<HistoryServiceGameScore>()) {
if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) { if (!result.link && result.cursor == HistoryInTextCursorState && g.contains(point)) {
result.link = gamescore->lnk; result.link = gamescore->lnk;

View File

@ -218,9 +218,9 @@ HistoryTextState Gif::getState(
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) { if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) {
if (_delete && rtlpoint(point, _width).x() >= _width - st::stickerPanDeleteIconBg.width() && point.y() < st::stickerPanDeleteIconBg.height()) { if (_delete && rtlpoint(point, _width).x() >= _width - st::stickerPanDeleteIconBg.width() && point.y() < st::stickerPanDeleteIconBg.height()) {
return _delete; return { nullptr, _delete };
} else { } else {
return _send; return { nullptr, _send };
} }
} }
return {}; return {};
@ -411,7 +411,7 @@ HistoryTextState Sticker::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) { if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) {
return _send; return { nullptr, _send };
} }
return {}; return {};
} }
@ -501,7 +501,7 @@ HistoryTextState Photo::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) { if (QRect(0, 0, _width, st::inlineMediaHeight).contains(point)) {
return _send; return { nullptr, _send };
} }
return {}; return {};
} }
@ -646,10 +646,10 @@ HistoryTextState Video::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) { if (QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) {
return _link; return { nullptr, _link };
} }
if (QRect(st::inlineThumbSize + st::inlineThumbSkip, 0, _width - st::inlineThumbSize - st::inlineThumbSkip, _height).contains(point)) { if (QRect(st::inlineThumbSize + st::inlineThumbSkip, 0, _width - st::inlineThumbSize - st::inlineThumbSkip, _height).contains(point)) {
return _send; return { nullptr, _send };
} }
return {}; return {};
} }
@ -787,11 +787,11 @@ HistoryTextState File::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (QRect(0, st::inlineRowMargin, st::msgFileSize, st::msgFileSize).contains(point)) { if (QRect(0, st::inlineRowMargin, st::msgFileSize, st::msgFileSize).contains(point)) {
return getShownDocument()->loading() ? _cancel : _open; return { nullptr, getShownDocument()->loading() ? _cancel : _open };
} else { } else {
auto left = st::msgFileSize + st::inlineThumbSkip; auto left = st::msgFileSize + st::inlineThumbSkip;
if (QRect(left, 0, _width - left, _height).contains(point)) { if (QRect(left, 0, _width - left, _height).contains(point)) {
return _send; return { nullptr, _send };
} }
} }
return {}; return {};
@ -955,7 +955,7 @@ HistoryTextState Contact::getState(
if (!QRect(0, st::inlineRowMargin, st::msgFileSize, st::inlineThumbSize).contains(point)) { if (!QRect(0, st::inlineRowMargin, st::msgFileSize, st::inlineThumbSize).contains(point)) {
auto left = (st::msgFileSize + st::inlineThumbSkip); auto left = (st::msgFileSize + st::inlineThumbSkip);
if (QRect(left, 0, _width - left, _height).contains(point)) { if (QRect(left, 0, _width - left, _height).contains(point)) {
return _send; return { nullptr, _send };
} }
} }
return {}; return {};
@ -1090,7 +1090,7 @@ HistoryTextState Article::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (_withThumb && QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) { if (_withThumb && QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) {
return _link; return { nullptr, _link };
} }
auto left = _withThumb ? (st::inlineThumbSize + st::inlineThumbSkip) : 0; auto left = _withThumb ? (st::inlineThumbSize + st::inlineThumbSkip) : 0;
if (QRect(left, 0, _width - left, _height).contains(point)) { if (QRect(left, 0, _width - left, _height).contains(point)) {
@ -1100,10 +1100,10 @@ HistoryTextState Article::getState(
auto descriptionLines = 2; auto descriptionLines = 2;
auto descriptionHeight = qMin(_description.countHeight(_width - left), st::normalFont->height * descriptionLines); auto descriptionHeight = qMin(_description.countHeight(_width - left), st::normalFont->height * descriptionLines);
if (rtlrect(left, st::inlineRowMargin + titleHeight + descriptionHeight, _urlWidth, st::normalFont->height, _width).contains(point)) { if (rtlrect(left, st::inlineRowMargin + titleHeight + descriptionHeight, _urlWidth, st::normalFont->height, _width).contains(point)) {
return _url; return { nullptr, _url };
} }
} }
return _send; return { nullptr, _send };
} }
return {}; return {};
} }
@ -1275,23 +1275,23 @@ HistoryTextState Game::getState(
HistoryStateRequest request) const { HistoryStateRequest request) const {
int left = st::inlineThumbSize + st::inlineThumbSkip; int left = st::inlineThumbSize + st::inlineThumbSkip;
if (QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) { if (QRect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize).contains(point)) {
return _send; return { nullptr, _send };
} }
if (QRect(left, 0, _width - left, _height).contains(point)) { if (QRect(left, 0, _width - left, _height).contains(point)) {
return _send; return { nullptr, _send };
} }
return {}; return {};
} }
void Game::prepareThumb(int width, int height) const { void Game::prepareThumb(int width, int height) const {
auto thumb = ([this]() { auto thumb = [this] {
if (auto photo = getResultPhoto()) { if (auto photo = getResultPhoto()) {
return photo->medium; return photo->medium;
} else if (auto document = getResultDocument()) { } else if (auto document = getResultDocument()) {
return document->thumb; return document->thumb;
} }
return ImagePtr(); return ImagePtr();
})(); }();
if (thumb->isNull()) { if (thumb->isNull()) {
return; return;
} }

View File

@ -24,6 +24,40 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
constexpr auto FullSelection = TextSelection { 0xFFFF, 0xFFFF }; constexpr auto FullSelection = TextSelection { 0xFFFF, 0xFFFF };
inline bool IsSubGroupSelection(TextSelection selection) {
return (selection.from == 0xFFFF) && (selection.to != 0xFFFF);
}
inline bool IsGroupItemSelection(
TextSelection selection,
int index) {
Expects(index >= 0 && index < 0x0F);
return IsSubGroupSelection(selection) && (selection.to & (1 << index));
}
inline [[nodiscard]] TextSelection AddGroupItemSelection(
TextSelection selection,
int index) {
Expects(index >= 0 && index < 0x0F);
const auto bit = uint16(1U << index);
return TextSelection(
0xFFFF,
IsSubGroupSelection(selection) ? (selection.to | bit) : bit);
}
inline[[nodiscard]] TextSelection RemoveGroupItemSelection(
TextSelection selection,
int index) {
Expects(index >= 0 && index < 0x0F);
const auto bit = uint16(1U << index);
return IsSubGroupSelection(selection)
? TextSelection(0xFFFF, selection.to & ~bit)
: selection;
}
extern TextParseOptions _textNameOptions, _textDlgOptions; extern TextParseOptions _textNameOptions, _textDlgOptions;
extern TextParseOptions _historyTextOptions, _historyBotOptions, _historyTextNoMonoOptions, _historyBotNoMonoOptions; extern TextParseOptions _historyTextOptions, _historyBotOptions, _historyTextNoMonoOptions, _historyBotNoMonoOptions;

View File

@ -201,9 +201,11 @@ bool Messenger::hideMediaView() {
return false; return false;
} }
void Messenger::showPhoto(not_null<const PhotoOpenClickHandler*> link, HistoryItem *item) { void Messenger::showPhoto(not_null<const PhotoOpenClickHandler*> link) {
return (!item && link->peer()) const auto item = App::histItemById(link->context());
? showPhoto(link->photo(), link->peer()) const auto peer = link->peer();
return (!item && peer)
? showPhoto(link->photo(), peer)
: showPhoto(link->photo(), item); : showPhoto(link->photo(), item);
} }
@ -214,7 +216,9 @@ void Messenger::showPhoto(not_null<PhotoData*> photo, HistoryItem *item) {
_mediaView->setFocus(); _mediaView->setFocus();
} }
void Messenger::showPhoto(not_null<PhotoData*> photo, PeerData *peer) { void Messenger::showPhoto(
not_null<PhotoData*> photo,
not_null<PeerData*> peer) {
if (_mediaView->isHidden()) Ui::hideLayer(anim::type::instant); if (_mediaView->isHidden()) Ui::hideLayer(anim::type::instant);
_mediaView->showPhoto(photo, peer); _mediaView->showPhoto(photo, peer);
_mediaView->activateWindow(); _mediaView->activateWindow();

View File

@ -86,9 +86,9 @@ public:
// MediaView interface. // MediaView interface.
void checkMediaViewActivation(); void checkMediaViewActivation();
bool hideMediaView(); bool hideMediaView();
void showPhoto(not_null<const PhotoOpenClickHandler*> link, HistoryItem *item = nullptr); void showPhoto(not_null<const PhotoOpenClickHandler*> link);
void showPhoto(not_null<PhotoData*> photo, HistoryItem *item); void showPhoto(not_null<PhotoData*> photo, HistoryItem *item);
void showPhoto(not_null<PhotoData*> photo, PeerData *item); void showPhoto(not_null<PhotoData*> photo, not_null<PeerData*> item);
void showDocument(not_null<DocumentData*> document, HistoryItem *item); void showDocument(not_null<DocumentData*> document, HistoryItem *item);
PeerData *ui_getPeerForMouseAction(); PeerData *ui_getPeerForMouseAction();

View File

@ -283,7 +283,7 @@ Photo::Photo(
not_null<PhotoData*> photo) not_null<PhotoData*> photo)
: ItemBase(parent) : ItemBase(parent)
, _data(photo) , _data(photo)
, _link(MakeShared<PhotoOpenClickHandler>(photo)) { , _link(MakeShared<PhotoOpenClickHandler>(photo, parent->fullId())) {
} }
void Photo::initDimensions() { void Photo::initDimensions() {
@ -352,7 +352,7 @@ HistoryTextState Photo::getState(
QPoint point, QPoint point,
HistoryStateRequest request) const { HistoryStateRequest request) const {
if (hasPoint(point)) { if (hasPoint(point)) {
return _link; return { parent(), _link };
} }
return {}; return {};
} }
@ -508,7 +508,12 @@ HistoryTextState Video::getState(
bool loaded = _data->loaded(); bool loaded = _data->loaded();
if (hasPoint(point)) { if (hasPoint(point)) {
return loaded ? _openl : (_data->loading() ? _cancell : _savel); const auto link = loaded
? _openl
: _data->loading()
? _cancell
: _savel;
return { parent(), link };
} }
return {}; return {};
} }
@ -687,13 +692,14 @@ HistoryTextState Voice::getState(
_st.songThumbSize, _st.songThumbSize,
_width); _width);
if (inner.contains(point)) { if (inner.contains(point)) {
return loaded const auto link = loaded
? _openl ? _openl
: ((_data->loading() || _data->status == FileUploading) : (_data->loading() || _data->status == FileUploading)
? _cancell ? _cancell
: _openl); : _openl;
return { parent(), link };
} }
auto result = HistoryTextState(); auto result = HistoryTextState(parent());
const auto statusmaxwidth = _width - nameleft - nameright; const auto statusmaxwidth = _width - nameleft - nameright;
const auto statusrect = rtlrect( const auto statusrect = rtlrect(
nameleft, nameleft,
@ -718,7 +724,7 @@ HistoryTextState Voice::getState(
st::normalFont->height, st::normalFont->height,
_width); _width);
if (namerect.contains(point) && !result.link && !_data->loading()) { if (namerect.contains(point) && !result.link && !_data->loading()) {
return _namel; return { parent(), _namel };
} }
return result; return result;
} }
@ -1014,11 +1020,12 @@ HistoryTextState Document::getState(
_st.songThumbSize, _st.songThumbSize,
_width); _width);
if (inner.contains(point)) { if (inner.contains(point)) {
return loaded const auto link = loaded
? _openl ? _openl
: ((_data->loading() || _data->status == FileUploading) : (_data->loading() || _data->status == FileUploading)
? _cancell ? _cancell
: _openl); : _openl;
return { parent(), link };
} }
const auto namerect = rtlrect( const auto namerect = rtlrect(
nameleft, nameleft,
@ -1027,7 +1034,7 @@ HistoryTextState Document::getState(
st::semiboldFont->height, st::semiboldFont->height,
_width); _width);
if (namerect.contains(point) && !_data->loading()) { if (namerect.contains(point) && !_data->loading()) {
return _namel; return { parent(), _namel };
} }
} else { } else {
const auto nameleft = _st.fileThumbSize + _st.filePadding.right(); const auto nameleft = _st.fileThumbSize + _st.filePadding.right();
@ -1047,11 +1054,12 @@ HistoryTextState Document::getState(
_width); _width);
if (rthumb.contains(point)) { if (rthumb.contains(point)) {
return loaded const auto link = loaded
? _openl ? _openl
: ((_data->loading() || _data->status == FileUploading) : (_data->loading() || _data->status == FileUploading)
? _cancell ? _cancell
: _savel); : _savel;
return { parent(), link };
} }
if (_data->status != FileUploadFailed) { if (_data->status != FileUploadFailed) {
@ -1062,7 +1070,7 @@ HistoryTextState Document::getState(
st::normalFont->height, st::normalFont->height,
_width); _width);
if (daterect.contains(point)) { if (daterect.contains(point)) {
return _msgl; return { parent(), _msgl };
} }
} }
if (!_data->loading() && _data->isValid()) { if (!_data->loading() && _data->isValid()) {
@ -1073,7 +1081,7 @@ HistoryTextState Document::getState(
_height - st::linksBorder, _height - st::linksBorder,
_width); _width);
if (loaded && leftofnamerect.contains(point)) { if (loaded && leftofnamerect.contains(point)) {
return _namel; return { parent(), _namel };
} }
const auto namerect = rtlrect( const auto namerect = rtlrect(
nameleft, nameleft,
@ -1082,7 +1090,7 @@ HistoryTextState Document::getState(
st::semiboldFont->height, st::semiboldFont->height,
_width); _width);
if (namerect.contains(point)) { if (namerect.contains(point)) {
return _namel; return { parent(), _namel };
} }
} }
} }
@ -1212,7 +1220,9 @@ Link::Link(
if (_page->type == WebPageProfile || _page->type == WebPageVideo) { if (_page->type == WebPageProfile || _page->type == WebPageVideo) {
_photol = MakeShared<UrlClickHandler>(_page->url); _photol = MakeShared<UrlClickHandler>(_page->url);
} else if (_page->type == WebPagePhoto || _page->siteName == qstr("Twitter") || _page->siteName == qstr("Facebook")) { } else if (_page->type == WebPagePhoto || _page->siteName == qstr("Twitter") || _page->siteName == qstr("Facebook")) {
_photol = MakeShared<PhotoOpenClickHandler>(_page->photo); _photol = MakeShared<PhotoOpenClickHandler>(
_page->photo,
parent->fullId());
} else { } else {
_photol = MakeShared<UrlClickHandler>(_page->url); _photol = MakeShared<UrlClickHandler>(_page->url);
} }
@ -1415,7 +1425,7 @@ HistoryTextState Link::getState(
HistoryStateRequest request) const { HistoryStateRequest request) const {
int32 left = st::linksPhotoSize + st::linksPhotoPadding, top = st::linksMargin.top() + st::linksBorder, w = _width - left; int32 left = st::linksPhotoSize + st::linksPhotoPadding, top = st::linksMargin.top() + st::linksBorder, w = _width - left;
if (rtlrect(0, top, st::linksPhotoSize, st::linksPhotoSize, _width).contains(point)) { if (rtlrect(0, top, st::linksPhotoSize, st::linksPhotoSize, _width).contains(point)) {
return _photol; return { parent(), _photol };
} }
if (!_title.isEmpty() && _text.isEmpty() && _links.size() == 1) { if (!_title.isEmpty() && _text.isEmpty() && _links.size() == 1) {
@ -1423,7 +1433,7 @@ HistoryTextState Link::getState(
} }
if (!_title.isEmpty()) { if (!_title.isEmpty()) {
if (rtlrect(left, top, qMin(w, _titlew), st::semiboldFont->height, _width).contains(point)) { if (rtlrect(left, top, qMin(w, _titlew), st::semiboldFont->height, _width).contains(point)) {
return _photol; return { parent(), _photol };
} }
top += st::webPageTitleFont->height; top += st::webPageTitleFont->height;
} }
@ -1432,7 +1442,7 @@ HistoryTextState Link::getState(
} }
for (int32 i = 0, l = _links.size(); i < l; ++i) { for (int32 i = 0, l = _links.size(); i < l; ++i) {
if (rtlrect(left, top, qMin(w, _links.at(i).width), st::normalFont->height, _width).contains(point)) { if (rtlrect(left, top, qMin(w, _links.at(i).width), st::normalFont->height, _width).contains(point)) {
return ClickHandlerPtr(_links[i].lnk); return { parent(), ClickHandlerPtr(_links[i].lnk) };
} }
top += st::normalFont->height; top += st::normalFont->height;
} }

View File

@ -94,6 +94,8 @@ CoverWidget::CoverWidget(QWidget *parent, UserData *self)
} }
PhotoData *CoverWidget::validatePhoto() const { PhotoData *CoverWidget::validatePhoto() const {
Expects(_self != nullptr);
const auto photo = _self->userpicPhotoId() const auto photo = _self->userpicPhotoId()
? App::photo(_self->userpicPhotoId()) ? App::photo(_self->userpicPhotoId())
: nullptr; : nullptr;
@ -106,7 +108,7 @@ PhotoData *CoverWidget::validatePhoto() const {
} }
void CoverWidget::showPhoto() { void CoverWidget::showPhoto() {
if (auto photo = validatePhoto()) { if (const auto photo = validatePhoto()) {
Messenger::Instance().showPhoto(photo, _self); Messenger::Instance().showPhoto(photo, _self);
} }
} }