diff --git a/Telegram/Resources/style.txt b/Telegram/Resources/style.txt index 89fadeb4b4..5361d5fb1d 100644 --- a/Telegram/Resources/style.txt +++ b/Telegram/Resources/style.txt @@ -1991,11 +1991,6 @@ stickerPreviewDuration: 150; stickerPreviewBg: #FFFFFFB0; stickerPreviewMin: 0.1; -inlineResultsLeft: 15px; -inlineResultsSkip: 3px; -inlineMediaHeight: 96px; -inlineResultsMinWidth: 64px; - verifiedCheckProfile: sprite(285px, 235px, 18px, 18px); verifiedCheckProfilePos: point(7px, 6px); verifiedCheck: sprite(285px, 221px, 14px, 14px); @@ -2346,9 +2341,21 @@ linksMaxWidth: 520px; linksLetterFont: font(24px); linksMargin: 5px; linksBorder: 1px; -linksBorderColor: #eaeaea; +linksBorderFg: #eaeaea; linksDateColor: #000; linksDateMargin: 15px; linksPhotoCheck: sprite(184px, 196px, 16px, 16px); linksPhotoChecked: sprite(168px, 196px, 16px, 16px); + +inlineResultsLeft: 15px; +inlineResultsSkip: 3px; +inlineMediaHeight: 96px; +inlineThumbSize: 64px; +inlineThumbSkip: 10px; +inlineDescriptionFg: #8a8a8a; +inlineRowMargin: 6px; +inlineRowBorder: linksBorder; +inlineRowBorderFg: linksBorderFg; +inlineResultsMinWidth: 64px; +inlineDurationMargin: 3px; diff --git a/Telegram/SourceFiles/dropdown.cpp b/Telegram/SourceFiles/dropdown.cpp index 2e738e16a8..82baaddb08 100644 --- a/Telegram/SourceFiles/dropdown.cpp +++ b/Telegram/SourceFiles/dropdown.cpp @@ -1216,6 +1216,7 @@ StickerPanInner::StickerPanInner() : TWidget() , _showingSavedGifs(cShowingSavedGifs()) , _showingInlineItems(_showingSavedGifs) , _lastScrolled(0) +, _inlineWithThumb(false) , _selected(-1) , _pressedSel(-1) , _settings(this, lang(lng_stickers_you_have)) @@ -1301,7 +1302,7 @@ void StickerPanInner::paintEvent(QPaintEvent *e) { } void StickerPanInner::paintInlineItems(Painter &p, const QRect &r) { - InlinePaintContext context(getms(), false, _previewShown); + InlinePaintContext context(getms(), false, _previewShown, false); int32 top = st::emojiPanHeader; int32 fromx = rtl() ? (width() - r.x() - r.width()) : r.x(), tox = rtl() ? (width() - r.x()) : (r.x() + r.width()); @@ -1310,6 +1311,7 @@ void StickerPanInner::paintInlineItems(Painter &p, const QRect &r) { if (top >= r.top() + r.height()) break; if (top + inlineRow.height > r.top()) { int32 left = st::inlineResultsLeft; + if (row == rows - 1) context.lastRow = true; for (int32 col = 0, cols = inlineRow.items.size(); col < cols; ++col) { if (left >= tox) break; @@ -1750,9 +1752,9 @@ LayoutInlineItem *StickerPanInner::layoutPrepareInlineResult(InlineResult *resul } else if (result->type == qstr("photo")) { layout = new LayoutInlinePhoto(result, 0); } else if (result->type == qstr("web_player_video")) { - // layout = new LayoutInlineWebVideo(result, 0); + layout = new LayoutInlineWebVideo(result); } else if (result->type == qstr("article")) { -// layout = new LayoutInlineArticle(result, 0); + layout = new LayoutInlineArticle(result, _inlineWithThumb); } if (!layout) return 0; @@ -1927,6 +1929,16 @@ void StickerPanInner::refreshInlineRows(UserData *bot, const InlineResults &resu } _inlineRows.resize(untilrow); + if (_inlineRows.isEmpty()) { + _inlineWithThumb = false; + for (int32 i = until; i < count; ++i) { + if (!results.at(i)->thumb->isNull()) { + _inlineWithThumb = true; + break; + } + } + } + _inlineRows.reserve(count); InlineRow row; row.items.reserve(SavedGifsMaxPerRow); @@ -3072,6 +3084,10 @@ void EmojiPan::step_appearance(float64 ms, bool timer) { void EmojiPan::hideStart() { if (_removingSetId || s_inner.inlineResultsShown()) return; + hideAnimated(); +} + +void EmojiPan::hideAnimated() { if (_cache.isNull()) { QPixmap from = _fromCache, to = _toCache; _fromCache = _toCache = QPixmap(); @@ -3412,7 +3428,7 @@ void EmojiPan::onDelayedHide() { void EmojiPan::inlineBotChanged() { if (!_inlineBot) return; - if (!isHidden()) hideStart(); + if (!isHidden()) hideAnimated(); if (_inlineRequestId) MTP::cancel(_inlineRequestId); _inlineRequestId = 0; @@ -3572,7 +3588,7 @@ void EmojiPan::showInlineRows(bool newResults) { s_inner.refreshInlineRows(_inlineBot, clear ? InlineResults() : i.value()->results, false); if (newResults) s_scroll.scrollToY(0); if (clear && !isHidden() && _stickersShown && s_inner.inlineResultsShown()) { - hideStart(); + hideAnimated(); } else if (!clear) { _hideTimer.stop(); if (!isHidden() || _hiding) { diff --git a/Telegram/SourceFiles/dropdown.h b/Telegram/SourceFiles/dropdown.h index 22fb4cdc98..2387cc094c 100644 --- a/Telegram/SourceFiles/dropdown.h +++ b/Telegram/SourceFiles/dropdown.h @@ -430,6 +430,7 @@ private: QString _inlineBotTitle; uint64 _lastScrolled; QTimer _updateInlineItems; + bool _inlineWithThumb; typedef QVector<LayoutInlineItem*> InlineItems; struct InlineRow { @@ -617,6 +618,7 @@ private: bool _horizontal; void leaveToChildEvent(QEvent *e); + void hideAnimated(); void updateSelected(); void updateIcons(); diff --git a/Telegram/SourceFiles/gui/text.cpp b/Telegram/SourceFiles/gui/text.cpp index 042783aded..fb8401019e 100644 --- a/Telegram/SourceFiles/gui/text.cpp +++ b/Telegram/SourceFiles/gui/text.cpp @@ -1031,6 +1031,9 @@ public: _y = top; _yFrom = yFrom + top; _yTo = (yTo < 0) ? -1 : (yTo + top); + if (_elideLast) { + _yToElide = _yTo; + } _selectedFrom = selectedFrom; _selectedTo = selectedTo; _wLeft = _w = w; @@ -1081,7 +1084,7 @@ public: last_rBearing = _rb; last_rPadding = b->f_rpadding(); _wLeft = _w - (b->f_width() - last_rBearing); - if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yTo)) { + if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yToElide)) { _wLeft -= _elideRemoveFromEnd; } @@ -1135,7 +1138,7 @@ public: } int32 elidedLineHeight = qMax(_lineHeight, blockHeight); - bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yTo); + bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yToElide); if (elidedLine) { _lineHeight = elidedLineHeight; } else if (f != j) { @@ -1154,7 +1157,7 @@ public: last_rBearing = j->f_rbearing(); last_rPadding = j->rpadding; _wLeft = _w - (j_width - last_rBearing); - if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yTo)) { + if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yToElide)) { _wLeft -= _elideRemoveFromEnd; } @@ -1165,7 +1168,7 @@ public: } if (lpadding > 0) { // no words in this block, spaces only int32 elidedLineHeight = qMax(_lineHeight, blockHeight); - bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yTo); + bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yToElide); if (elidedLine) { _lineHeight = elidedLineHeight; } @@ -1179,7 +1182,7 @@ public: last_rBearing = _rb; last_rPadding = b->rpadding(); _wLeft = _w; - if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yTo)) { + if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yToElide)) { _wLeft -= _elideRemoveFromEnd; } @@ -1189,7 +1192,7 @@ public: } int32 elidedLineHeight = qMax(_lineHeight, blockHeight); - bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yTo); + bool elidedLine = _elideLast && (_y + elidedLineHeight >= _yToElide); if (elidedLine) { _lineHeight = elidedLineHeight; } @@ -1202,7 +1205,7 @@ public: last_rBearing = _rb; last_rPadding = b->f_rpadding(); _wLeft = _w - (b->f_width() - last_rBearing); - if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yTo)) { + if (_elideLast && _elideRemoveFromEnd > 0 && (_y + blockHeight >= _yToElide)) { _wLeft -= _elideRemoveFromEnd; } @@ -1301,7 +1304,7 @@ public: } ITextBlock *_endBlock = (_endBlockIter == _end) ? 0 : (*_endBlockIter); - bool elidedLine = _elideLast && _endBlock && (_y + _lineHeight >= _yTo); + bool elidedLine = _elideLast && _endBlock && (_y + _lineHeight >= _yToElide); int blockIndex = _lineStartBlock; ITextBlock *currentBlock = _t->_blocks[blockIndex]; @@ -2409,7 +2412,7 @@ private: int32 _elideRemoveFromEnd; style::align _align; QPen _originalPen; - int32 _yFrom, _yTo; + int32 _yFrom, _yTo, _yToElide; uint16 _selectedFrom, _selectedTo; const QChar *_str; diff --git a/Telegram/SourceFiles/historywidget.cpp b/Telegram/SourceFiles/historywidget.cpp index fb060ed341..ec177cdd21 100644 --- a/Telegram/SourceFiles/historywidget.cpp +++ b/Telegram/SourceFiles/historywidget.cpp @@ -4345,9 +4345,8 @@ void HistoryWidget::onSend(bool ctrlShiftEnter, MsgId replyTo) { _saveDraftStart = getms(); onDraftSave(); - if (!_attachMention.isHidden()) _attachMention.hideStart(); + onCheckMentionDropdown(); if (!_attachType.isHidden()) _attachType.hideStart(); - if (!_emojiPan.isHidden()) _emojiPan.hideStart(); if (replyTo < 0) cancelReply(lastKeyboardUsed); if (_previewData && _previewData->pendingTill) previewCancel(); @@ -6357,6 +6356,7 @@ void HistoryWidget::onInlineResultSend(InlineResult *result, UserData *bot) { _history->addNewMessage(MTP_message(MTP_int(flags), MTP_int(newId.msg), MTP_int(fromChannelName ? 0 : MTP::authedId()), peerToMTP(_history->peer->id), MTPPeer(), MTPint(), MTP_int(bot ? peerToUser(bot->id) : 0), MTP_int(replyToId()), MTP_int(unixtime()), MTP_string(""), MTP_messageMediaPhoto(photo, MTP_string(result->caption)), MTPnullMarkup, MTPnullEntities, MTP_int(1)), NewMessageUnread); } } else { + flags |= MTPDmessage::flag_entities; if (result->noWebPage) { sendFlags |= MTPmessages_SendMessage::flag_no_webpage; } @@ -6373,9 +6373,8 @@ void HistoryWidget::onInlineResultSend(InlineResult *result, UserData *bot) { _saveDraftStart = getms(); onDraftSave(); - if (!_attachMention.isHidden()) _attachMention.hideStart(); + onCheckMentionDropdown(); if (!_attachType.isHidden()) _attachType.hideStart(); - if (!_emojiPan.isHidden()) _emojiPan.hideStart(); _field.setFocus(); } diff --git a/Telegram/SourceFiles/layout.cpp b/Telegram/SourceFiles/layout.cpp index 9b84b33a03..8b4dce32cc 100644 --- a/Telegram/SourceFiles/layout.cpp +++ b/Telegram/SourceFiles/layout.cpp @@ -824,7 +824,7 @@ void LayoutOverviewDocument::paint(Painter &p, const QRect &clip, uint32 selecti QRect shadow(rtlrect(nameleft, 0, _width - nameleft, st::linksBorder, _width)); if (clip.intersects(shadow)) { - p.fillRect(clip.intersected(shadow), st::linksBorderColor); + p.fillRect(clip.intersected(shadow), st::linksBorderFg); } QRect rthumb(rtlrect(0, st::linksBorder + st::msgFileThumbPadding.top(), st::msgFileThumbSize, st::msgFileThumbSize, _width)); @@ -1242,7 +1242,7 @@ void LayoutOverviewLink::paint(Painter &p, const QRect &clip, uint32 selection, } if (clip.intersects(rtlrect(left, 0, w, st::linksBorder, _width))) { - p.fillRect(clip.intersected(rtlrect(left, 0, w, st::linksBorder, _width)), st::linksBorderColor); + p.fillRect(clip.intersected(rtlrect(left, 0, w, st::linksBorder, _width)), st::linksBorderFg); } } @@ -1394,7 +1394,7 @@ void LayoutInlineGif::paint(Painter &p, const QRect &clip, uint32 selection, con QRect r(0, 0, _width, height); if (animating) { - if (!_thumb.isNull()) const_cast<LayoutInlineGif*>(this)->_thumb = QPixmap(); + if (!_thumb.isNull()) _thumb = QPixmap(); const InlinePaintContext *ctx = context->toInlinePaintContext(); t_assert(ctx); p.drawPixmap(r.topLeft(), _gif->current(frame.width(), frame.height(), _width, height, ctx->paused ? 0 : context->ms)); @@ -1532,7 +1532,7 @@ void LayoutInlineGif::prepareThumb(int32 width, int32 height, const QSize &frame if (_doc && !_doc->thumb->isNull()) { if (_doc->thumb->loaded()) { if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) { - const_cast<LayoutInlineGif*>(this)->_thumb = _doc->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height); + _thumb = _doc->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height); } } else { _doc->thumb->load(); @@ -1540,7 +1540,7 @@ void LayoutInlineGif::prepareThumb(int32 width, int32 height, const QSize &frame } else if (_result && !_result->thumb_url.isEmpty()) { if (_result->thumb->loaded()) { if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) { - const_cast<LayoutInlineGif*>(this)->_thumb = _result->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height); + _thumb = _result->thumb->pixNoCache(frame.width() * cIntRetinaFactor(), frame.height() * cIntRetinaFactor(), true, false, false, width, height); } } else { _result->thumb->load(); @@ -1804,3 +1804,233 @@ void LayoutInlinePhoto::content_forget() { _result->forget(); } } + +LayoutInlineWebVideo::LayoutInlineWebVideo(InlineResult *result) : LayoutInlineItem(result, 0, 0) +, _send(new SendInlineItemLink()) +, _title(st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - st::inlineThumbSize - st::inlineThumbSkip) +, _description(st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - st::inlineThumbSize - st::inlineThumbSkip) { + if (_result->duration) { + _duration = formatDurationText(_result->duration); + } +} + +void LayoutInlineWebVideo::initDimensions() { + _maxw = st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft; + int32 textWidth = _maxw - (_result->thumb->isNull() ? 0 : (st::inlineThumbSize + st::inlineThumbSkip)); + TextParseOptions titleOpts = { 0, _maxw, 2 * st::semiboldFont->height, Qt::LayoutDirectionAuto }; + _title.setText(st::semiboldFont, textOneLine(_result->title), titleOpts); + int32 titleHeight = qMin(_title.countHeight(_maxw), 2 * st::semiboldFont->height); + + int32 descriptionLines = _result->thumb->isNull() ? 3 : (titleHeight > st::semiboldFont->height ? 1 : 2); + + TextParseOptions descriptionOpts = { TextParseMultiline, _maxw, descriptionLines * st::normalFont->height, Qt::LayoutDirectionAuto }; + _description.setText(st::normalFont, _result->description, descriptionOpts); + int32 descriptionHeight = qMin(_description.countHeight(_maxw), descriptionLines * st::normalFont->height); + + _minh = _result->thumb->isNull() ? titleHeight + descriptionHeight : st::inlineThumbSize; + _minh += st::inlineRowMargin * 2 + st::inlineRowBorder; +} + +void LayoutInlineWebVideo::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const { + int32 left = 0; + if (!_result->thumb->isNull()) { + left = st::inlineThumbSize + st::inlineThumbSkip; + prepareThumb(st::inlineThumbSize, st::inlineThumbSize); + if (_thumb.isNull()) { + p.fillRect(rtlrect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize, _width), _result->thumb->isNull() ? st::black : st::overviewPhotoBg); + } else { + p.drawPixmapLeft(0, st::inlineRowMargin, _width, _thumb); + } + + if (!_duration.isEmpty()) { + int32 durationTop = st::inlineRowMargin + st::inlineThumbSize - st::normalFont->height - st::inlineDurationMargin; + p.fillRect(rtlrect(0, durationTop - st::inlineDurationMargin, st::inlineThumbSize, st::normalFont->height + 2 * st::inlineDurationMargin, _width), st::msgDateImgBg); + p.setPen(st::white); + p.setFont(st::normalFont); + p.drawTextRight(_width - st::inlineThumbSize + st::inlineDurationMargin, durationTop, _width, _duration); + } + } + + p.setPen(st::black); + _title.drawLeftElided(p, left, st::inlineRowMargin, _width - left, _width, 2); + int32 titleHeight = qMin(_title.countHeight(_width - left), st::semiboldFont->height * 2); + + p.setPen(st::inlineDescriptionFg); + int32 descriptionLines = _result->thumb->isNull() ? 3 : (titleHeight > st::semiboldFont->height ? 1 : 2); + _description.drawLeftElided(p, left, st::inlineRowMargin + titleHeight, _width - left, _width, descriptionLines); + + const InlinePaintContext *ctx = context->toInlinePaintContext(); + t_assert(ctx); + if (!ctx->lastRow) { + p.fillRect(rtlrect(left, _height - st::inlineRowBorder, _width - left, st::inlineRowBorder, _width), st::inlineRowBorderFg); + } +} + +void LayoutInlineWebVideo::getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const { + if (x >= 0 && x < _width && y >= 0 && y < _height) { + link = _send; + } +} + +void LayoutInlineWebVideo::prepareThumb(int32 width, int32 height) const { + if (_result->thumb->loaded()) { + if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) { + int32 w = qMax(_result->thumb->width(), 1), h = qMax(_result->thumb->height(), 1); + if (w * height > h * width) { + if (height < h) { + w = w * height / h; + h = height; + } + } else { + if (width < w) { + h = h * width / w; + w = width; + } + } + _thumb = _result->thumb->pixNoCache(w * cIntRetinaFactor(), h * cIntRetinaFactor(), true, false, false, width, height); + } + } else { + _result->thumb->load(); + } +} + +LayoutInlineArticle::LayoutInlineArticle(InlineResult *result, bool withThumb) : LayoutInlineItem(result, 0, 0) +, _send(new SendInlineItemLink()) +, _url(result->url.isEmpty() ? 0 : linkFromUrl(result->url)) +, _withThumb(withThumb) +, _title(st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - st::inlineThumbSize - st::inlineThumbSkip) +, _description(st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft - st::inlineThumbSize - st::inlineThumbSkip) { + QVector<QStringRef> parts = _result->url.splitRef('/'); + if (!parts.isEmpty()) { + QStringRef domain = parts.at(0); + if (parts.size() > 2 && domain.endsWith(':') && parts.at(1).isEmpty()) { // http:// and others + domain = parts.at(2); + } + + parts = domain.split('@').back().split('.'); + if (parts.size() > 1) { + _letter = parts.at(parts.size() - 2).at(0).toUpper(); + } + } + if (_letter.isEmpty() && !_result->title.isEmpty()) { + _letter = _result->title.at(0).toUpper(); + } +} + +void LayoutInlineArticle::initDimensions() { + _maxw = st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft; + int32 textWidth = _maxw - (_withThumb ? (st::inlineThumbSize + st::inlineThumbSkip) : 0); + TextParseOptions titleOpts = { 0, _maxw, 2 * st::semiboldFont->height, Qt::LayoutDirectionAuto }; + _title.setText(st::semiboldFont, textOneLine(_result->title), titleOpts); + int32 titleHeight = qMin(_title.countHeight(_maxw), 2 * st::semiboldFont->height); + + int32 descriptionLines = (_withThumb || _url) ? 2 : 3; + + TextParseOptions descriptionOpts = { TextParseMultiline, _maxw, descriptionLines * st::normalFont->height, Qt::LayoutDirectionAuto }; + _description.setText(st::normalFont, _result->description, descriptionOpts); + int32 descriptionHeight = qMin(_description.countHeight(_maxw), descriptionLines * st::normalFont->height); + + _minh = titleHeight + descriptionHeight; + if (_url) _minh += st::normalFont->height; + if (_withThumb) _minh = qMax(_minh, int32(st::inlineThumbSize)); + _minh += st::inlineRowMargin * 2 + st::inlineRowBorder; +} + +int32 LayoutInlineArticle::resizeGetHeight(int32 width) { + _width = qMin(width, _maxw); + if (_url) { + _urlText = _result->url; + _urlWidth = st::normalFont->width(_urlText); + if (_urlWidth > _width - st::inlineThumbSize - st::inlineThumbSkip) { + _urlText = st::normalFont->elided(_result->url, _width - st::inlineThumbSize - st::inlineThumbSkip); + _urlWidth = st::normalFont->width(_urlText); + } + } + _height = _minh; + return _height; +} + +void LayoutInlineArticle::paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const { + int32 left = 0; + if (_withThumb) { + left = st::inlineThumbSize + st::inlineThumbSkip; + prepareThumb(st::inlineThumbSize, st::inlineThumbSize); + QRect rthumb(rtlrect(0, st::inlineRowMargin, st::inlineThumbSize, st::inlineThumbSize, _width)); + if (_thumb.isNull()) { + if (_result->thumb->isNull() && !_letter.isEmpty()) { + int32 index = (_letter.at(0).unicode() % 4); + style::color colors[] = { st::msgFileRedColor, st::msgFileYellowColor, st::msgFileGreenColor, st::msgFileBlueColor }; + + p.fillRect(rthumb, colors[index]); + if (!_letter.isEmpty()) { + p.setFont(st::linksLetterFont); + p.setPen(st::white); + p.drawText(rthumb, _letter, style::al_center); + } + } else { + p.fillRect(rthumb, st::overviewPhotoBg); + } + } else { + p.drawPixmapLeft(rthumb.topLeft(), _width, _thumb); + } + } + + p.setPen(st::black); + _title.drawLeftElided(p, left, st::inlineRowMargin, _width - left, _width, 2); + int32 titleHeight = qMin(_title.countHeight(_width - left), st::semiboldFont->height * 2); + + p.setPen(st::inlineDescriptionFg); + int32 descriptionLines = (_withThumb || _url) ? 2 : 3; + _description.drawLeftElided(p, left, st::inlineRowMargin + titleHeight, _width - left, _width, descriptionLines); + + if (_url) { + int32 descriptionHeight = qMin(_description.countHeight(_width - left), st::normalFont->height * descriptionLines); + p.drawTextLeft(left, st::inlineRowMargin + titleHeight + descriptionHeight, _width, _urlText, _urlWidth); + } + + const InlinePaintContext *ctx = context->toInlinePaintContext(); + t_assert(ctx); + if (!ctx->lastRow) { + p.fillRect(rtlrect(left, _height - st::inlineRowBorder, _width - left, st::inlineRowBorder, _width), st::inlineRowBorderFg); + } +} + +void LayoutInlineArticle::getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const { + if (x >= 0 && x < _width && y >= 0 && y < _height) { + if (_url) { + int32 left = st::inlineThumbSize + st::inlineThumbSkip; + int32 titleHeight = qMin(_title.countHeight(_width - left), st::semiboldFont->height * 2); + int32 descriptionLines = 2; + int32 descriptionHeight = qMin(_description.countHeight(_width - left), st::normalFont->height * descriptionLines); + if (rtlrect(left, st::inlineRowMargin + titleHeight + descriptionHeight, _urlWidth, st::normalFont->height, _width).contains(x, y)) { + link = _url; + return; + } + } + link = _send; + } +} + +void LayoutInlineArticle::prepareThumb(int32 width, int32 height) const { + if (_result->thumb->isNull()) return; + + if (_result->thumb->loaded()) { + if (_thumb.width() != width * cIntRetinaFactor() || _thumb.height() != height * cIntRetinaFactor()) { + int32 w = qMax(_result->thumb->width(), 1), h = qMax(_result->thumb->height(), 1); + if (w * height > h * width) { + if (height < h) { + w = w * height / h; + h = height; + } + } else { + if (width < w) { + h = h * width / w; + w = width; + } + } + _thumb = _result->thumb->pixNoCache(w * cIntRetinaFactor(), h * cIntRetinaFactor(), true, false, false, width, height); + } + } else { + _result->thumb->load(); + } +} diff --git a/Telegram/SourceFiles/layout.h b/Telegram/SourceFiles/layout.h index 74150081b2..b069067382 100644 --- a/Telegram/SourceFiles/layout.h +++ b/Telegram/SourceFiles/layout.h @@ -477,12 +477,15 @@ private: class InlinePaintContext : public PaintContext { public: - InlinePaintContext(uint64 ms, bool selecting, bool paused) : PaintContext(ms, selecting), paused(paused) { + InlinePaintContext(uint64 ms, bool selecting, bool paused, bool lastRow) + : PaintContext(ms, selecting) + , paused(paused) + , lastRow(lastRow) { } virtual const InlinePaintContext *toInlinePaintContext() const { return this; } - bool paused; + bool paused, lastRow; }; class LayoutInlineItem : public LayoutItem { @@ -629,3 +632,49 @@ private: void prepareThumb(int32 width, int32 height, const QSize &frame) const; }; + +class LayoutInlineWebVideo : public LayoutInlineItem { +public: + LayoutInlineWebVideo(InlineResult *result); + + virtual void initDimensions(); + + virtual void paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const; + virtual void getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const; + +private: + + TextLinkPtr _send; + + mutable QPixmap _thumb; + Text _title, _description; + QString _duration; + int32 _durationWidth; + + void prepareThumb(int32 width, int32 height) const; + +}; + +class LayoutInlineArticle : public LayoutInlineItem { +public: + LayoutInlineArticle(InlineResult *result, bool withThumb); + + virtual void initDimensions(); + virtual int32 resizeGetHeight(int32 width); + + virtual void paint(Painter &p, const QRect &clip, uint32 selection, const PaintContext *context) const; + virtual void getState(TextLinkPtr &link, HistoryCursorState &cursor, int32 x, int32 y) const; + +private: + + TextLinkPtr _send, _url; + + bool _withThumb; + mutable QPixmap _thumb; + Text _title, _description; + QString _letter, _urlText; + int32 _urlWidth; + + void prepareThumb(int32 width, int32 height) const; + +};