Add search in GIFs to EmojiPanel.

This commit is contained in:
John Preston 2017-03-31 18:50:02 +03:00
parent 0690c4f98c
commit e8ed307278
18 changed files with 480 additions and 478 deletions

View File

@ -774,13 +774,11 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
"lng_recent_stickers" = "Frequently used"; "lng_recent_stickers" = "Frequently used";
"lng_switch_stickers" = "Stickers"; "lng_switch_stickers" = "Stickers";
"lng_switch_stickers_gifs" = "GIFs & Stickers";
"lng_switch_emoji" = "Emoji"; "lng_switch_emoji" = "Emoji";
"lng_switch_gifs" = "GIFs"; "lng_switch_gifs" = "GIFs";
"lng_stickers_featured_add" = "Add"; "lng_stickers_featured_add" = "Add";
"lng_gifs_search" = "Search GIFs";
"lng_saved_gifs" = "Saved GIFs";
"lng_inline_bot_results" = "Results from {inline_bot}";
"lng_inline_bot_no_results" = "No results"; "lng_inline_bot_no_results" = "No results";
"lng_inline_bot_via" = "via {inline_bot}"; "lng_inline_bot_via" = "via {inline_bot}";

View File

@ -1138,7 +1138,7 @@ namespace {
if (saved.size() > Global::SavedGifsLimit()) saved.pop_back(); if (saved.size() > Global::SavedGifsLimit()) saved.pop_back();
Local::writeSavedGifs(); Local::writeSavedGifs();
if (App::main()) emit App::main()->savedGifsUpdated(); AuthSession::Current().data().savedGifsUpdated().notify();
cSetLastSavedGifsUpdate(0); cSetLastSavedGifsUpdate(0);
App::main()->updateStickers(); App::main()->updateStickers();
} }
@ -1146,8 +1146,8 @@ namespace {
void checkSavedGif(HistoryItem *item) { void checkSavedGif(HistoryItem *item) {
if (!item->Has<HistoryMessageForwarded>() && (item->out() || item->history()->peer == App::self())) { if (!item->Has<HistoryMessageForwarded>() && (item->out() || item->history()->peer == App::self())) {
if (HistoryMedia *media = item->getMedia()) { if (auto media = item->getMedia()) {
if (DocumentData *doc = media->getDocument()) { if (auto doc = media->getDocument()) {
if (doc->isGifv()) { if (doc->isGifv()) {
addSavedGif(doc); addSavedGif(doc);
} }

View File

@ -47,6 +47,9 @@ public:
base::Observable<void> &moreChatsLoaded() { base::Observable<void> &moreChatsLoaded() {
return _moreChatsLoaded; return _moreChatsLoaded;
} }
base::Observable<void> &savedGifsUpdated() {
return _savedGifsUpdated;
}
void copyFrom(const AuthSessionData &other) { void copyFrom(const AuthSessionData &other) {
_variables = other._variables; _variables = other._variables;
@ -76,6 +79,7 @@ private:
base::Variable<bool> _contactsLoaded = { false }; base::Variable<bool> _contactsLoaded = { false };
base::Variable<bool> _allChatsLoaded = { false }; base::Variable<bool> _allChatsLoaded = { false };
base::Observable<void> _moreChatsLoaded; base::Observable<void> _moreChatsLoaded;
base::Observable<void> _savedGifsUpdated;
Variables _variables; Variables _variables;
}; };

View File

@ -237,6 +237,43 @@ contactsAboutFg: windowSubTextFgOver;
contactsAboutTop: 60px; contactsAboutTop: 60px;
contactsAboutBottom: 19px; contactsAboutBottom: 19px;
contactsSearchField: InputField(defaultInputField) {
textBg: transparent;
textMargins: margins(2px, 7px, 2px, 0px);
placeholderFg: placeholderFg;
placeholderFgActive: placeholderFgActive;
placeholderFgError: placeholderFgActive;
placeholderMargins: margins(2px, 0px, 2px, 0px);
placeholderScale: 0.;
placeholderFont: normalFont;
border: 0px;
borderActive: 0px;
heightMin: 32px;
font: normalFont;
}
contactsSearchCancel: CrossButton {
width: 44px;
height: 44px;
cross: CrossAnimation {
size: 36px;
skip: 12px;
stroke: 2px;
minScale: 0.3;
}
crossFg: boxTitleCloseFg;
crossFgOver: boxTitleCloseFgOver;
crossPosition: point(4px, 4px);
duration: 150;
ripple: RippleAnimation(defaultRippleAnimation) {
color: windowBgOver;
}
}
contactsMultiSelect: MultiSelect { contactsMultiSelect: MultiSelect {
bg: boxSearchBg; bg: boxSearchBg;
padding: margins(8px, 6px, 8px, 6px); padding: margins(8px, 6px, 8px, 6px);
@ -271,47 +308,12 @@ contactsMultiSelect: MultiSelect {
} }
itemSkip: 8px; itemSkip: 8px;
field: InputField(defaultInputField) { field: contactsSearchField;
textBg: transparent;
textMargins: margins(2px, 7px, 2px, 0px);
placeholderFg: placeholderFg;
placeholderFgActive: placeholderFgActive;
placeholderFgError: placeholderFgActive;
placeholderMargins: margins(2px, 0px, 2px, 0px);
placeholderScale: 0.;
placeholderFont: normalFont;
border: 0px;
borderActive: 0px;
heightMin: 32px;
font: normalFont;
}
fieldMinWidth: 42px; fieldMinWidth: 42px;
fieldIcon: boxFieldSearchIcon; fieldIcon: boxFieldSearchIcon;
fieldIconSkip: 36px; fieldIconSkip: 36px;
fieldCancel: CrossButton { fieldCancel: contactsSearchCancel;
width: 44px;
height: 44px;
cross: CrossAnimation {
size: 36px;
skip: 12px;
stroke: 2px;
minScale: 0.3;
}
crossFg: boxTitleCloseFg;
crossFgOver: boxTitleCloseFgOver;
crossPosition: point(4px, 4px);
duration: 150;
ripple: RippleAnimation(defaultRippleAnimation) {
color: windowBgOver;
}
}
fieldCancelSkip: 40px; fieldCancelSkip: 40px;
} }
contactsPhotoCheckbox: RoundImageCheckbox { contactsPhotoCheckbox: RoundImageCheckbox {

View File

@ -79,7 +79,6 @@ enum {
ClipThreadsCount = 8, ClipThreadsCount = 8,
AverageGifSize = 320 * 240, AverageGifSize = 320 * 240,
WaitBeforeGifPause = 200, // wait 200ms for gif draw before pausing it WaitBeforeGifPause = 200, // wait 200ms for gif draw before pausing it
InlineBotRequestDelay = 400, // wait 400ms before context bot realtime request
RecentInlineBotsLimit = 10, RecentInlineBotsLimit = 10,
AVBlockSize = 4096, // 4Kb for ffmpeg blocksize AVBlockSize = 4096, // 4Kb for ffmpeg blocksize

View File

@ -3360,10 +3360,8 @@ HistoryWidget::HistoryWidget(QWidget *parent, gsl::not_null<Window::Controller*>
void HistoryWidget::start() { void HistoryWidget::start() {
connect(App::main(), SIGNAL(stickersUpdated()), this, SLOT(onStickersUpdated())); connect(App::main(), SIGNAL(stickersUpdated()), this, SLOT(onStickersUpdated()));
connect(App::main(), SIGNAL(savedGifsUpdated()), _emojiPanel, SLOT(refreshSavedGifs()));
updateRecentStickers(); updateRecentStickers();
if (App::main()) emit App::main()->savedGifsUpdated(); AuthSession::Current().data().savedGifsUpdated().notify();
connect(App::api(), SIGNAL(fullPeerUpdated(PeerData*)), this, SLOT(onFullPeerUpdated(PeerData*))); connect(App::api(), SIGNAL(fullPeerUpdated(PeerData*)), this, SLOT(onFullPeerUpdated(PeerData*)));
} }
@ -4176,7 +4174,7 @@ void HistoryWidget::savedGifsGot(const MTPmessages_SavedGifs &gifs) {
Local::writeSavedGifs(); Local::writeSavedGifs();
if (App::main()) emit App::main()->savedGifsUpdated(); AuthSession::Current().data().savedGifsUpdated().notify();
} }
void HistoryWidget::saveGif(DocumentData *doc) { void HistoryWidget::saveGif(DocumentData *doc) {
@ -4430,6 +4428,7 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re
_peer = App::peer(peerId); _peer = App::peer(peerId);
_channel = peerToChannel(_peer->id); _channel = peerToChannel(_peer->id);
_canSendMessages = canSendMessages(_peer); _canSendMessages = canSendMessages(_peer);
_emojiPanel->setInlineQueryPeer(_peer);
} }
updateTopBarSelection(); updateTopBarSelection();

View File

@ -30,7 +30,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "media/player/media_player_instance.h" #include "media/player/media_player_instance.h"
#include "history/history_location_manager.h" #include "history/history_location_manager.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "mainwidget.h" #include "auth_session.h"
#include "lang.h" #include "lang.h"
namespace InlineBots { namespace InlineBots {
@ -122,14 +122,14 @@ void Gif::setPosition(int32 position) {
} }
void DeleteSavedGifClickHandler::onClickImpl() const { void DeleteSavedGifClickHandler::onClickImpl() const {
int32 index = cSavedGifs().indexOf(_data); auto index = cSavedGifs().indexOf(_data);
if (index >= 0) { if (index >= 0) {
cRefSavedGifs().remove(index); cRefSavedGifs().remove(index);
Local::writeSavedGifs(); Local::writeSavedGifs();
MTP::send(MTPmessages_SaveGif(_data->mtpInput(), MTP_bool(true))); MTP::send(MTPmessages_SaveGif(_data->mtpInput(), MTP_bool(true)));
} }
if (App::main()) emit App::main()->savedGifsUpdated(); AuthSession::Current().data().savedGifsUpdated().notify();
} }
void Gif::paint(Painter &p, const QRect &clip, const PaintContext *context) const { void Gif::paint(Painter &p, const QRect &clip, const PaintContext *context) const {

View File

@ -39,6 +39,11 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
namespace InlineBots { namespace InlineBots {
namespace Layout { namespace Layout {
namespace internal { namespace internal {
namespace {
constexpr auto kInlineBotRequestDelay = 400;
} // namespace
Inner::Inner(QWidget *parent) : TWidget(parent) { Inner::Inner(QWidget *parent) : TWidget(parent) {
setMaxHeight(st::emojiPanMaxHeight - st::emojiCategory.height); setMaxHeight(st::emojiPanMaxHeight - st::emojiCategory.height);
@ -413,7 +418,6 @@ int Inner::refreshInlineRows(UserData *bot, const CacheEntry *entry, bool result
clearSelection(); clearSelection();
t_assert(_inlineBot != 0); t_assert(_inlineBot != 0);
_inlineBotTitle = lng_inline_bot_results(lt_inline_bot, _inlineBot->username.isEmpty() ? _inlineBot->name : ('@' + _inlineBot->username));
auto count = int(entry->results.size()); auto count = int(entry->results.size());
auto from = validateExistingInlineRows(entry->results); auto from = validateExistingInlineRows(entry->results);
@ -1048,7 +1052,7 @@ void Widget::queryInlineBot(UserData *bot, PeerData *peer, QString query) {
showInlineRows(true); showInlineRows(true);
} else { } else {
_inlineNextQuery = query; _inlineNextQuery = query;
_inlineRequestTimer.start(InlineBotRequestDelay); _inlineRequestTimer.start(internal::kInlineBotRequestDelay);
} }
} }
} }

View File

@ -120,7 +120,6 @@ private:
int _visibleBottom = 0; int _visibleBottom = 0;
UserData *_inlineBot; UserData *_inlineBot;
QString _inlineBotTitle;
TimeMs _lastScrolled = 0; TimeMs _lastScrolled = 0;
QTimer _updateInlineItems; QTimer _updateInlineItems;
bool _inlineWithThumb = false; bool _inlineWithThumb = false;

View File

@ -413,7 +413,6 @@ signals:
void dialogRowReplaced(Dialogs::Row *oldRow, Dialogs::Row *newRow); void dialogRowReplaced(Dialogs::Row *oldRow, Dialogs::Row *newRow);
void dialogsUpdated(); void dialogsUpdated();
void stickersUpdated(); void stickersUpdated();
void savedGifsUpdated();
public slots: public slots:
void webPagesOrGamesUpdate(); void webPagesOrGamesUpdate();

View File

@ -364,11 +364,11 @@ private:
template <typename Request, typename, typename> template <typename Request, typename, typename>
Sender::SpecificRequestBuilder<Request> Sender::request(Request &&request) noexcept { Sender::SpecificRequestBuilder<Request> Sender::request(Request &&request) noexcept {
return SpecificRequestBuilder<Request>(this, std::move(request)); return SpecificRequestBuilder<Request>(this, std::move(request));
} }
inline Sender::SentRequestWrap Sender::request(mtpRequestId requestId) noexcept { inline Sender::SentRequestWrap Sender::request(mtpRequestId requestId) noexcept {
return SentRequestWrap(this, requestId); return SentRequestWrap(this, requestId);
} }
} // namespace MTP } // namespace MTP

View File

@ -28,7 +28,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/widgets/shadow.h" #include "ui/widgets/shadow.h"
#include "ui/widgets/discrete_sliders.h" #include "ui/widgets/discrete_sliders.h"
#include "ui/widgets/scroll_area.h" #include "ui/widgets/scroll_area.h"
#include "inline_bots/inline_bot_result.h"
#include "stickers/stickers.h" #include "stickers/stickers.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "lang.h" #include "lang.h"
@ -332,16 +331,14 @@ EmojiPanel::EmojiPanel(QWidget *parent) : TWidget(parent)
connect(stickers(), SIGNAL(checkForHide()), this, SLOT(onCheckForHide())); connect(stickers(), SIGNAL(checkForHide()), this, SLOT(onCheckForHide()));
connect(gifs(), SIGNAL(selected(DocumentData*)), this, SIGNAL(stickerSelected(DocumentData*))); connect(gifs(), SIGNAL(selected(DocumentData*)), this, SIGNAL(stickerSelected(DocumentData*)));
connect(gifs(), SIGNAL(selected(PhotoData*)), this, SIGNAL(photoSelected(PhotoData*))); connect(gifs(), SIGNAL(selected(PhotoData*)), this, SIGNAL(photoSelected(PhotoData*)));
connect(gifs(), SIGNAL(selected(InlineBots::Result*, UserData*)), this, SIGNAL(inlineResultSelected(InlineBots::Result*, UserData*))); connect(gifs(), SIGNAL(selected(InlineBots::Result*,UserData*)), this, SIGNAL(inlineResultSelected(InlineBots::Result*,UserData*)));
connect(gifs(), SIGNAL(emptyInlineRows()), this, SLOT(onEmptyInlineRows())); connect(gifs(), &GifsListWidget::cancelled, this, [this] {
hideAnimated();
});
_saveConfigTimer.setSingleShot(true); _saveConfigTimer.setSingleShot(true);
connect(&_saveConfigTimer, SIGNAL(timeout()), this, SLOT(onSaveConfig())); connect(&_saveConfigTimer, SIGNAL(timeout()), this, SLOT(onSaveConfig()));
// inline bots
_inlineRequestTimer.setSingleShot(true);
connect(&_inlineRequestTimer, SIGNAL(timeout()), this, SLOT(onInlineRequest()));
if (cPlatform() == dbipMac || cPlatform() == dbipMacOld) { if (cPlatform() == dbipMac || cPlatform() == dbipMacOld) {
connect(App::wnd()->windowHandle(), SIGNAL(activeChanged()), this, SLOT(onWndActiveChanged())); connect(App::wnd()->windowHandle(), SIGNAL(activeChanged()), this, SLOT(onWndActiveChanged()));
} }
@ -436,6 +433,7 @@ void EmojiPanel::paintEvent(QPaintEvent *e) {
_showAnimation.reset(); _showAnimation.reset();
if (!switching && !opacityAnimating) { if (!switching && !opacityAnimating) {
showAll(); showAll();
currentTab()->widget()->afterShown();
} }
} }
@ -455,6 +453,7 @@ void EmojiPanel::paintEvent(QPaintEvent *e) {
_slideAnimation.reset(); _slideAnimation.reset();
if (!opacityAnimating) { if (!opacityAnimating) {
showAll(); showAll();
currentTab()->widget()->afterShown();
} }
InvokeQueued(this, [this] { InvokeQueued(this, [this] {
if (_hideAfterSlide && !_a_slide.animating()) { if (_hideAfterSlide && !_a_slide.animating()) {
@ -567,15 +566,6 @@ void EmojiPanel::refreshStickers() {
if (isHidden() || _currentTabType != TabType::Stickers) { if (isHidden() || _currentTabType != TabType::Stickers) {
stickers()->preloadImages(); stickers()->preloadImages();
} }
update();
}
void EmojiPanel::refreshSavedGifs() {
gifs()->refreshSavedGifs();
if (isHidden() || _currentTabType != TabType::Gifs) {
gifs()->preloadImages();
}
update();
} }
void EmojiPanel::opacityAnimationCallback() { void EmojiPanel::opacityAnimationCallback() {
@ -586,6 +576,7 @@ void EmojiPanel::opacityAnimationCallback() {
hideFinished(); hideFinished();
} else if (!_a_show.animating() && !_a_slide.animating()) { } else if (!_a_show.animating() && !_a_slide.animating()) {
showAll(); showAll();
currentTab()->widget()->afterShown();
} }
} }
} }
@ -613,6 +604,9 @@ void EmojiPanel::prepareCache() {
} }
void EmojiPanel::startOpacityAnimation(bool hiding) { void EmojiPanel::startOpacityAnimation(bool hiding) {
if (!_scroll->isHidden()) {
currentTab()->widget()->beforeHiding();
}
_hiding = false; _hiding = false;
prepareCache(); prepareCache();
_hiding = hiding; _hiding = hiding;
@ -747,6 +741,10 @@ void EmojiPanel::stickersInstalled(uint64 setId) {
showAnimated(); showAnimated();
} }
void EmojiPanel::setInlineQueryPeer(PeerData *peer) {
gifs()->setInlineQueryPeer(peer);
}
bool EmojiPanel::ui_isInlineItemBeingChosen() { bool EmojiPanel::ui_isInlineItemBeingChosen() {
return (_currentTabType == TabType::Gifs && !isHidden()); return (_currentTabType == TabType::Gifs && !isHidden());
} }
@ -816,6 +814,10 @@ void EmojiPanel::switchTab() {
auto wasTab = _currentTabType; auto wasTab = _currentTabType;
currentTab()->saveScrollTop(); currentTab()->saveScrollTop();
if (!_scroll->isHidden()) {
currentTab()->widget()->beforeHiding();
}
auto wasCache = grabForComplexAnimation(GrabType::Slide); auto wasCache = grabForComplexAnimation(GrabType::Slide);
auto widget = _scroll->takeWidget<Inner>(); auto widget = _scroll->takeWidget<Inner>();
@ -828,6 +830,8 @@ void EmojiPanel::switchTab() {
if (_currentTabType == TabType::Gifs) { if (_currentTabType == TabType::Gifs) {
Notify::clipStopperHidden(ClipStopperSavedGifsPanel); Notify::clipStopperHidden(ClipStopperSavedGifsPanel);
} }
currentTab()->widget()->refreshRecent();
currentTab()->widget()->preloadImages();
setWidgetToScrollArea(); setWidgetToScrollArea();
auto nowCache = grabForComplexAnimation(GrabType::Slide); auto nowCache = grabForComplexAnimation(GrabType::Slide);
@ -883,10 +887,6 @@ void EmojiPanel::onCheckForHide() {
} }
} }
void EmojiPanel::clearInlineBot() {
inlineBotChanged();
}
bool EmojiPanel::overlaps(const QRect &globalRect) const { bool EmojiPanel::overlaps(const QRect &globalRect) const {
if (isHidden() || !_cache.isNull()) return false; if (isHidden() || !_cache.isNull()) return false;
@ -896,147 +896,6 @@ bool EmojiPanel::overlaps(const QRect &globalRect) const {
|| inner.marginsRemoved(QMargins(0, st::buttonRadius, 0, st::buttonRadius)).contains(testRect); || inner.marginsRemoved(QMargins(0, st::buttonRadius, 0, st::buttonRadius)).contains(testRect);
} }
void EmojiPanel::inlineBotChanged() {
if (!_inlineBot) return;
if (!isHidden() && !_hiding) {
if (!rect().contains(mapFromGlobal(QCursor::pos()))) {
hideAnimated();
}
}
if (_inlineRequestId) MTP::cancel(_inlineRequestId);
_inlineRequestId = 0;
_inlineQuery = _inlineNextQuery = _inlineNextOffset = QString();
_inlineBot = nullptr;
_inlineCache.clear();
gifs()->inlineBotChanged();
gifs()->hideInlineRowsPanel();
Notify::inlineBotRequesting(false);
}
void EmojiPanel::inlineResultsDone(const MTPmessages_BotResults &result) {
_inlineRequestId = 0;
Notify::inlineBotRequesting(false);
auto it = _inlineCache.find(_inlineQuery);
auto adding = (it != _inlineCache.cend());
if (result.type() == mtpc_messages_botResults) {
auto &d = result.c_messages_botResults();
auto &v = d.vresults.v;
auto queryId = d.vquery_id.v;
if (it == _inlineCache.cend()) {
it = _inlineCache.emplace(_inlineQuery, std::make_unique<InlineCacheEntry>()).first;
}
auto entry = it->second.get();
entry->nextOffset = qs(d.vnext_offset);
if (d.has_switch_pm() && d.vswitch_pm.type() == mtpc_inlineBotSwitchPM) {
auto &switchPm = d.vswitch_pm.c_inlineBotSwitchPM();
entry->switchPmText = qs(switchPm.vtext);
entry->switchPmStartToken = qs(switchPm.vstart_param);
}
if (auto count = v.size()) {
entry->results.reserve(entry->results.size() + count);
}
auto added = 0;
for_const (const auto &res, v) {
if (auto result = InlineBots::Result::create(queryId, res)) {
++added;
entry->results.push_back(std::move(result));
}
}
if (!added) {
entry->nextOffset = QString();
}
} else if (adding) {
it->second->nextOffset = QString();
}
if (!showInlineRows(!adding)) {
it->second->nextOffset = QString();
}
onScroll();
}
void EmojiPanel::queryInlineBot(UserData *bot, PeerData *peer, QString query) {
bool force = false;
_inlineQueryPeer = peer;
if (bot != _inlineBot) {
inlineBotChanged();
_inlineBot = bot;
force = true;
//if (_inlineBot->isBotInlineGeo()) {
// Ui::show(Box<InformBox>(lang(lng_bot_inline_geo_unavailable)));
//}
}
//if (_inlineBot && _inlineBot->isBotInlineGeo()) {
// return;
//}
if (_inlineQuery != query || force) {
if (_inlineRequestId) {
MTP::cancel(_inlineRequestId);
_inlineRequestId = 0;
Notify::inlineBotRequesting(false);
}
if (_inlineCache.find(query) != _inlineCache.cend()) {
_inlineRequestTimer.stop();
_inlineQuery = _inlineNextQuery = query;
showInlineRows(true);
} else {
_inlineNextQuery = query;
_inlineRequestTimer.start(InlineBotRequestDelay);
}
}
}
void EmojiPanel::onInlineRequest() {
if (_inlineRequestId || !_inlineBot || !_inlineQueryPeer) return;
_inlineQuery = _inlineNextQuery;
QString nextOffset;
auto it = _inlineCache.find(_inlineQuery);
if (it != _inlineCache.cend()) {
nextOffset = it->second->nextOffset;
if (nextOffset.isEmpty()) return;
}
Notify::inlineBotRequesting(true);
_inlineRequestId = request(MTPmessages_GetInlineBotResults(MTP_flags(0), _inlineBot->inputUser, _inlineQueryPeer->input, MTPInputGeoPoint(), MTP_string(_inlineQuery), MTP_string(nextOffset))).done([this](const MTPmessages_BotResults &result, mtpRequestId requestId) {
inlineResultsDone(result);
}).fail([this](const RPCError &error) {
// show error?
Notify::inlineBotRequesting(false);
_inlineRequestId = 0;
}).handleAllErrors().send();
}
void EmojiPanel::onEmptyInlineRows() {
if (!_inlineBot) {
gifs()->hideInlineRowsPanel();
} else {
gifs()->clearInlineRowsPanel();
}
}
bool EmojiPanel::refreshInlineRows(int32 *added) {
auto it = _inlineCache.find(_inlineQuery);
const InlineCacheEntry *entry = nullptr;
if (it != _inlineCache.cend()) {
if (!it->second->results.empty() || !it->second->switchPmText.isEmpty()) {
entry = it->second.get();
}
_inlineNextOffset = it->second->nextOffset;
}
if (!entry) prepareCache();
auto result = gifs()->refreshInlineRows(_inlineBot, entry, false);
if (added) *added = result;
return (entry != nullptr);
}
void EmojiPanel::scrollToY(int y) { void EmojiPanel::scrollToY(int y) {
_scroll->scrollToY(y); _scroll->scrollToY(y);
@ -1044,28 +903,6 @@ void EmojiPanel::scrollToY(int y) {
_topShadow->update(); _topShadow->update();
} }
int32 EmojiPanel::showInlineRows(bool newResults) {
int32 added = 0;
bool clear = !refreshInlineRows(&added);
if (newResults) {
scrollToY(0);
}
auto hidden = isHidden();
if (clear) {
if (!_hiding) {
_cache = QPixmap(); // clear after refreshInlineRows()
}
} else {
if (_currentTabType != TabType::Gifs) {
_tabsSlider->setActiveSection(static_cast<int>(TabType::Gifs));
}
showAnimated();
}
return added;
}
EmojiPanel::Inner::Inner(QWidget *parent) : TWidget(parent) { EmojiPanel::Inner::Inner(QWidget *parent) : TWidget(parent) {
} }

View File

@ -23,13 +23,9 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/twidget.h" #include "ui/twidget.h"
#include "ui/effects/panel_animation.h" #include "ui/effects/panel_animation.h"
#include "mtproto/sender.h" #include "mtproto/sender.h"
#include "inline_bots/inline_bot_layout_item.h"
#include "auth_session.h" #include "auth_session.h"
namespace InlineBots { namespace InlineBots {
namespace Layout {
class ItemBase;
} // namespace Layout
class Result; class Result;
} // namespace InlineBots } // namespace InlineBots
@ -46,17 +42,7 @@ class EmojiListWidget;
class StickersListWidget; class StickersListWidget;
class GifsListWidget; class GifsListWidget;
using InlineResult = InlineBots::Result; class EmojiPanel : public TWidget {
using InlineResults = std::vector<std::unique_ptr<InlineResult>>;
using InlineItem = InlineBots::Layout::ItemBase;
struct InlineCacheEntry {
QString nextOffset;
QString switchPmText, switchPmStartToken;
InlineResults results;
};
class EmojiPanel : public TWidget, private MTP::Sender {
Q_OBJECT Q_OBJECT
public: public:
@ -73,10 +59,8 @@ public:
void stickersInstalled(uint64 setId); void stickersInstalled(uint64 setId);
void queryInlineBot(UserData *bot, PeerData *peer, QString query);
void clearInlineBot();
bool overlaps(const QRect &globalRect) const; bool overlaps(const QRect &globalRect) const;
void setInlineQueryPeer(PeerData *peer);
bool ui_isInlineItemBeingChosen(); bool ui_isInlineItemBeingChosen();
@ -102,7 +86,6 @@ public slots:
private slots: private slots:
void hideByTimerOrLeave(); void hideByTimerOrLeave();
void refreshSavedGifs();
void onWndActiveChanged(); void onWndActiveChanged();
void onScroll(); void onScroll();
@ -112,9 +95,6 @@ private slots:
void onSaveConfig(); void onSaveConfig();
void onSaveConfigDelayed(int delay); void onSaveConfigDelayed(int delay);
void onInlineRequest();
void onEmptyInlineRows();
signals: signals:
void emojiSelected(EmojiPtr emoji); void emojiSelected(EmojiPtr emoji);
void stickerSelected(DocumentData *sticker); void stickerSelected(DocumentData *sticker);
@ -255,19 +235,6 @@ private:
QTimer _saveConfigTimer; QTimer _saveConfigTimer;
// inline bots
std::map<QString, std::unique_ptr<InlineCacheEntry>> _inlineCache;
QTimer _inlineRequestTimer;
void inlineBotChanged();
int32 showInlineRows(bool newResults);
bool refreshInlineRows(int32 *added = 0);
UserData *_inlineBot = nullptr;
PeerData *_inlineQueryPeer = nullptr;
QString _inlineQuery, _inlineNextQuery, _inlineNextOffset;
mtpRequestId _inlineRequestId = 0;
void inlineResultsDone(const MTPmessages_BotResults &result);
}; };
class EmojiPanel::Inner : public TWidget { class EmojiPanel::Inner : public TWidget {
@ -292,6 +259,11 @@ public:
void panelHideFinished(); void panelHideFinished();
virtual void clearSelection() = 0; virtual void clearSelection() = 0;
virtual void afterShown() {
}
virtual void beforeHiding() {
}
virtual object_ptr<InnerFooter> createFooter() = 0; virtual object_ptr<InnerFooter> createFooter() = 0;
signals: signals:

View File

@ -22,6 +22,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "styles/style_stickers.h" #include "styles/style_stickers.h"
#include "ui/widgets/buttons.h" #include "ui/widgets/buttons.h"
#include "ui/widgets/input_fields.h"
#include "ui/effects/ripple_animation.h" #include "ui/effects/ripple_animation.h"
#include "boxes/stickers_box.h" #include "boxes/stickers_box.h"
#include "inline_bots/inline_bot_result.h" #include "inline_bots/inline_bot_result.h"
@ -34,8 +35,10 @@ namespace ChatHelpers {
namespace { namespace {
constexpr auto kSaveChosenTabTimeout = 1000; constexpr auto kSaveChosenTabTimeout = 1000;
constexpr auto kSearchRequestDelay = 400;
constexpr auto kStickersPanelPerRow = Stickers::kPanelPerRow; constexpr auto kStickersPanelPerRow = Stickers::kPanelPerRow;
constexpr auto kInlineItemsMaxPerRow = 5; constexpr auto kInlineItemsMaxPerRow = 5;
constexpr auto kSearchBotUsername = "gif";
} // namespace } // namespace
@ -43,20 +46,67 @@ class GifsListWidget::Footer : public EmojiPanel::InnerFooter {
public: public:
Footer(gsl::not_null<GifsListWidget*> parent); Footer(gsl::not_null<GifsListWidget*> parent);
void stealFocus();
void returnFocus();
protected: protected:
void paintEvent(QPaintEvent *e) override;
void processPanelHideFinished() override; void processPanelHideFinished() override;
private: private:
gsl::not_null<GifsListWidget*> _pan; gsl::not_null<GifsListWidget*> _pan;
object_ptr<Ui::InputField> _field;
object_ptr<Ui::CrossButton> _cancel;
QPointer<QWidget> _focusTakenFrom;
}; };
GifsListWidget::Footer::Footer(gsl::not_null<GifsListWidget*> parent) : InnerFooter(parent) GifsListWidget::Footer::Footer(gsl::not_null<GifsListWidget*> parent) : InnerFooter(parent)
, _pan(parent) { , _pan(parent)
, _field(this, st::gifsSearchField, lang(lng_gifs_search))
, _cancel(this, st::gifsSearchCancel) {
_field->resize(width() - st::gifsSearchFieldPosition.x() - st::gifsSearchCancelPosition.x() - st::gifsSearchCancel.width, _field->height());
_field->moveToLeft(st::gifsSearchFieldPosition.x(), st::gifsSearchFieldPosition.y());
connect(_field, &Ui::InputField::submitted, this, [this](bool ctrlShiftEnter) {
_pan->sendInlineRequest();
});
connect(_field, &Ui::InputField::cancelled, this, [this] {
if (_field->getLastText().isEmpty()) {
emit _pan->cancelled();
} else {
_field->setText(QString());
}
});
connect(_field, &Ui::InputField::changed, this, [this] {
_cancel->toggleAnimated(!_field->getLastText().isEmpty());
_pan->searchForGifs(_field->getLastText());
});
_cancel->moveToRight(st::gifsSearchCancelPosition.x(), st::gifsSearchCancelPosition.y());
}
void GifsListWidget::Footer::stealFocus() {
if (!_focusTakenFrom) {
_focusTakenFrom = QApplication::focusWidget();
}
_field->setFocus();
}
void GifsListWidget::Footer::returnFocus() {
if (_focusTakenFrom) {
_focusTakenFrom->setFocus();
}
}
void GifsListWidget::Footer::paintEvent(QPaintEvent *e) {
Painter p(this);
st::gifsSearchIcon.paint(p, st::gifsSearchIconPosition.x(), st::gifsSearchIconPosition.y(), width());
} }
void GifsListWidget::Footer::processPanelHideFinished() { void GifsListWidget::Footer::processPanelHideFinished() {
// TODO Clear search _field->setText(QString());
} }
GifsListWidget::GifsListWidget(QWidget *parent) : Inner(parent) GifsListWidget::GifsListWidget(QWidget *parent) : Inner(parent)
@ -72,6 +122,12 @@ GifsListWidget::GifsListWidget(QWidget *parent) : Inner(parent)
_updateInlineItems.setSingleShot(true); _updateInlineItems.setSingleShot(true);
connect(&_updateInlineItems, SIGNAL(timeout()), this, SLOT(onUpdateInlineItems())); connect(&_updateInlineItems, SIGNAL(timeout()), this, SLOT(onUpdateInlineItems()));
_inlineRequestTimer.setSingleShot(true);
connect(&_inlineRequestTimer, &QTimer::timeout, this, [this] { sendInlineRequest(); });
subscribe(AuthSession::Current().data().savedGifsUpdated(), [this] {
refreshSavedGifs();
});
subscribe(AuthSession::CurrentDownloaderTaskFinished(), [this] { subscribe(AuthSession::CurrentDownloaderTaskFinished(), [this] {
update(); update();
}); });
@ -90,9 +146,13 @@ void GifsListWidget::setVisibleTopBottom(int visibleTop, int visibleBottom) {
if (top != getVisibleTop()) { if (top != getVisibleTop()) {
_lastScrolled = getms(); _lastScrolled = getms();
} }
auto visibleHeight = (visibleBottom - visibleTop); checkLoadMore();
if (visibleBottom + visibleHeight > height()) { }
// onInlineRequest(); // TODO
void GifsListWidget::checkLoadMore() {
auto visibleHeight = (getVisibleBottom() - getVisibleTop());
if (getVisibleBottom() + visibleHeight > height()) {
sendInlineRequest();
} }
} }
@ -103,11 +163,8 @@ int GifsListWidget::countHeight() {
} }
auto minimalLastHeight = (visibleHeight - st::stickerPanPadding); auto minimalLastHeight = (visibleHeight - st::stickerPanPadding);
auto result = st::stickerPanPadding; auto result = st::stickerPanPadding;
if (_switchPmButton) { for (int i = 0, l = _rows.count(); i < l; ++i) {
result += _switchPmButton->height() + st::inlineResultsSkip; result += _rows[i].height;
}
for (int i = 0, l = _inlineRows.count(); i < l; ++i) {
result += _inlineRows[i].height;
} }
return qMax(minimalLastHeight, result) + st::stickerPanPadding; return qMax(minimalLastHeight, result) + st::stickerPanPadding;
} }
@ -118,6 +175,56 @@ GifsListWidget::~GifsListWidget() {
deleteUnusedInlineLayouts(); deleteUnusedInlineLayouts();
} }
void GifsListWidget::cancelGifsSearch() {
if (_inlineRequestId) {
request(_inlineRequestId).cancel();
_inlineRequestId = 0;
}
_inlineRequestTimer.stop();
_inlineQuery = _inlineNextQuery = _inlineNextOffset = QString();
_inlineCache.clear();
refreshInlineRows(nullptr, true);
}
void GifsListWidget::inlineResultsDone(const MTPmessages_BotResults &result) {
_inlineRequestId = 0;
auto it = _inlineCache.find(_inlineQuery);
auto adding = (it != _inlineCache.cend());
if (result.type() == mtpc_messages_botResults) {
auto &d = result.c_messages_botResults();
auto &v = d.vresults.v;
auto queryId = d.vquery_id.v;
if (it == _inlineCache.cend()) {
it = _inlineCache.emplace(_inlineQuery, std::make_unique<InlineCacheEntry>()).first;
}
auto entry = it->second.get();
entry->nextOffset = qs(d.vnext_offset);
if (auto count = v.size()) {
entry->results.reserve(entry->results.size() + count);
}
auto added = 0;
for_const (const auto &res, v) {
if (auto result = InlineBots::Result::create(queryId, res)) {
++added;
entry->results.push_back(std::move(result));
}
}
if (!added) {
entry->nextOffset = QString();
}
} else if (adding) {
it->second->nextOffset = QString();
}
if (!showInlineRows(!adding)) {
it->second->nextOffset = QString();
}
checkLoadMore();
}
void GifsListWidget::paintEvent(QPaintEvent *e) { void GifsListWidget::paintEvent(QPaintEvent *e) {
Painter p(this); Painter p(this);
auto clip = e->rect(); auto clip = e->rect();
@ -127,7 +234,7 @@ void GifsListWidget::paintEvent(QPaintEvent *e) {
} }
void GifsListWidget::paintInlineItems(Painter &p, QRect clip) { void GifsListWidget::paintInlineItems(Painter &p, QRect clip) {
if (_inlineRows.isEmpty() && !_switchPmButton) { if (_rows.isEmpty()) {
p.setFont(st::normalFont); p.setFont(st::normalFont);
p.setPen(st::noContactsColor); p.setPen(st::noContactsColor);
p.drawText(QRect(0, 0, width(), (height() / 3) * 2 + st::normalFont->height), lang(lng_inline_bot_no_results), style::al_center); p.drawText(QRect(0, 0, width(), (height() / 3) * 2 + st::normalFont->height), lang(lng_inline_bot_no_results), style::al_center);
@ -137,14 +244,10 @@ void GifsListWidget::paintInlineItems(Painter &p, QRect clip) {
InlineBots::Layout::PaintContext context(getms(), false, gifPaused, false); InlineBots::Layout::PaintContext context(getms(), false, gifPaused, false);
auto top = st::stickerPanPadding; auto top = st::stickerPanPadding;
if (_switchPmButton) {
top += _switchPmButton->height() + st::inlineResultsSkip;
}
auto fromx = rtl() ? (width() - clip.x() - clip.width()) : clip.x(); auto fromx = rtl() ? (width() - clip.x() - clip.width()) : clip.x();
auto tox = rtl() ? (width() - clip.x()) : (clip.x() + clip.width()); auto tox = rtl() ? (width() - clip.x()) : (clip.x() + clip.width());
for (auto row = 0, rows = _inlineRows.size(); row != rows; ++row) { for (auto row = 0, rows = _rows.size(); row != rows; ++row) {
auto &inlineRow = _inlineRows[row]; auto &inlineRow = _rows[row];
if (top >= clip.top() + clip.height()) { if (top >= clip.top() + clip.height()) {
break; break;
} }
@ -210,11 +313,11 @@ void GifsListWidget::mouseReleaseEvent(QMouseEvent *e) {
} }
void GifsListWidget::selectInlineResult(int row, int column) { void GifsListWidget::selectInlineResult(int row, int column) {
if (row >= _inlineRows.size() || column >= _inlineRows.at(row).items.size()) { if (row >= _rows.size() || column >= _rows[row].items.size()) {
return; return;
} }
auto item = _inlineRows[row].items[column]; auto item = _rows[row].items[column];
if (auto photo = item->getPhoto()) { if (auto photo = item->getPhoto()) {
if (photo->medium->loaded() || photo->thumb->loaded()) { if (photo->medium->loaded() || photo->thumb->loaded()) {
emit selected(photo); emit selected(photo);
@ -232,7 +335,7 @@ void GifsListWidget::selectInlineResult(int row, int column) {
} }
} else if (auto inlineResult = item->getResult()) { } else if (auto inlineResult = item->getResult()) {
if (inlineResult->onChoose(item)) { if (inlineResult->onChoose(item)) {
emit selected(inlineResult, _inlineBot); emit selected(inlineResult, _searchBot);
} }
} }
} }
@ -258,8 +361,8 @@ void GifsListWidget::enterFromChildEvent(QEvent *e, QWidget *child) {
void GifsListWidget::clearSelection() { void GifsListWidget::clearSelection() {
if (_selected >= 0) { if (_selected >= 0) {
int srow = _selected / MatrixRowShift, scol = _selected % MatrixRowShift; int srow = _selected / MatrixRowShift, scol = _selected % MatrixRowShift;
t_assert(srow >= 0 && srow < _inlineRows.size() && scol >= 0 && scol < _inlineRows.at(srow).items.size()); t_assert(srow >= 0 && srow < _rows.size() && scol >= 0 && scol < _rows[srow].items.size());
ClickHandler::clearActive(_inlineRows.at(srow).items.at(scol)); ClickHandler::clearActive(_rows[srow].items[scol]);
setCursor(style::cur_default); setCursor(style::cur_default);
} }
_selected = _pressed = -1; _selected = _pressed = -1;
@ -295,18 +398,18 @@ void GifsListWidget::processPanelHideFinished() {
} }
} }
bool GifsListWidget::inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, InlineRow &row, int32 &sumWidth) { bool GifsListWidget::inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, Row &row, int32 &sumWidth) {
InlineItem *layout = nullptr; LayoutItem *layout = nullptr;
if (savedGif) { if (savedGif) {
layout = layoutPrepareSavedGif(savedGif, (_inlineRows.size() * MatrixRowShift) + row.items.size()); layout = layoutPrepareSavedGif(savedGif, (_rows.size() * MatrixRowShift) + row.items.size());
} else if (result) { } else if (result) {
layout = layoutPrepareInlineResult(result, (_inlineRows.size() * MatrixRowShift) + row.items.size()); layout = layoutPrepareInlineResult(result, (_rows.size() * MatrixRowShift) + row.items.size());
} }
if (!layout) return false; if (!layout) return false;
layout->preload(); layout->preload();
if (inlineRowFinalize(row, sumWidth, layout->isFullLine())) { if (inlineRowFinalize(row, sumWidth, layout->isFullLine())) {
layout->setPosition(_inlineRows.size() * MatrixRowShift); layout->setPosition(_rows.size() * MatrixRowShift);
} }
sumWidth += layout->maxWidth(); sumWidth += layout->maxWidth();
@ -318,14 +421,14 @@ bool GifsListWidget::inlineRowsAddItem(DocumentData *savedGif, InlineResult *res
return true; return true;
} }
bool GifsListWidget::inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force) { bool GifsListWidget::inlineRowFinalize(Row &row, int32 &sumWidth, bool force) {
if (row.items.isEmpty()) return false; if (row.items.isEmpty()) return false;
auto full = (row.items.size() >= kInlineItemsMaxPerRow); auto full = (row.items.size() >= kInlineItemsMaxPerRow);
auto big = (sumWidth >= st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft); auto big = (sumWidth >= st::emojiPanWidth - st::emojiScroll.width - st::inlineResultsLeft);
if (full || big || force) { if (full || big || force) {
_inlineRows.push_back(layoutInlineRow(row, (full || big) ? sumWidth : 0)); _rows.push_back(layoutInlineRow(row, (full || big) ? sumWidth : 0));
row = InlineRow(); row = Row();
row.items.reserve(kInlineItemsMaxPerRow); row.items.reserve(kInlineItemsMaxPerRow);
sumWidth = 0; sumWidth = 0;
return true; return true;
@ -339,8 +442,8 @@ void GifsListWidget::refreshSavedGifs() {
auto &saved = cSavedGifs(); auto &saved = cSavedGifs();
if (!saved.isEmpty()) { if (!saved.isEmpty()) {
_inlineRows.reserve(saved.size()); _rows.reserve(saved.size());
auto row = InlineRow(); auto row = Row();
row.items.reserve(kInlineItemsMaxPerRow); row.items.reserve(kInlineItemsMaxPerRow);
auto sumWidth = 0; auto sumWidth = 0;
for_const (auto &gif, saved) { for_const (auto &gif, saved) {
@ -350,16 +453,19 @@ void GifsListWidget::refreshSavedGifs() {
} }
deleteUnusedGifLayouts(); deleteUnusedGifLayouts();
int32 h = countHeight(); auto newHeight = countHeight();
if (h != height()) resize(width(), h); if (newHeight != height()) {
resize(width(), newHeight);
}
update(); update();
} }
updateSelected();
}
void GifsListWidget::inlineBotChanged() { if (isVisible()) {
refreshInlineRows(nullptr, nullptr, true); updateSelected();
} else {
preloadImages();
}
} }
void GifsListWidget::clearInlineRows(bool resultsDeleted) { void GifsListWidget::clearInlineRows(bool resultsDeleted) {
@ -367,19 +473,19 @@ void GifsListWidget::clearInlineRows(bool resultsDeleted) {
_selected = _pressed = -1; _selected = _pressed = -1;
} else { } else {
clearSelection(); clearSelection();
for_const (auto &row, _inlineRows) { for_const (auto &row, _rows) {
for_const (auto &item, row.items) { for_const (auto &item, row.items) {
item->setPosition(-1); item->setPosition(-1);
} }
} }
} }
_inlineRows.clear(); _rows.clear();
} }
InlineItem *GifsListWidget::layoutPrepareSavedGif(DocumentData *doc, int32 position) { GifsListWidget::LayoutItem *GifsListWidget::layoutPrepareSavedGif(DocumentData *doc, int32 position) {
auto it = _gifLayouts.find(doc); auto it = _gifLayouts.find(doc);
if (it == _gifLayouts.cend()) { if (it == _gifLayouts.cend()) {
if (auto layout = InlineItem::createLayoutGif(this, doc)) { if (auto layout = LayoutItem::createLayoutGif(this, doc)) {
it = _gifLayouts.emplace(doc, std::move(layout)).first; it = _gifLayouts.emplace(doc, std::move(layout)).first;
it->second->initDimensions(); it->second->initDimensions();
} else { } else {
@ -392,10 +498,10 @@ InlineItem *GifsListWidget::layoutPrepareSavedGif(DocumentData *doc, int32 posit
return it->second.get(); return it->second.get();
} }
InlineItem *GifsListWidget::layoutPrepareInlineResult(InlineResult *result, int32 position) { GifsListWidget::LayoutItem *GifsListWidget::layoutPrepareInlineResult(InlineResult *result, int32 position) {
auto it = _inlineLayouts.find(result); auto it = _inlineLayouts.find(result);
if (it == _inlineLayouts.cend()) { if (it == _inlineLayouts.cend()) {
if (auto layout = InlineItem::createLayout(this, result, _inlineWithThumb)) { if (auto layout = LayoutItem::createLayout(this, result, _inlineWithThumb)) {
it = _inlineLayouts.emplace(result, std::move(layout)).first; it = _inlineLayouts.emplace(result, std::move(layout)).first;
it->second->initDimensions(); it->second->initDimensions();
} else { } else {
@ -409,7 +515,7 @@ InlineItem *GifsListWidget::layoutPrepareInlineResult(InlineResult *result, int3
} }
void GifsListWidget::deleteUnusedGifLayouts() { void GifsListWidget::deleteUnusedGifLayouts() {
if (_inlineRows.isEmpty() || _section != Section::Gifs) { // delete all if (_rows.isEmpty() || _section != Section::Gifs) { // delete all
_gifLayouts.clear(); _gifLayouts.clear();
} else { } else {
for (auto i = _gifLayouts.begin(); i != _gifLayouts.cend();) { for (auto i = _gifLayouts.begin(); i != _gifLayouts.cend();) {
@ -423,7 +529,7 @@ void GifsListWidget::deleteUnusedGifLayouts() {
} }
void GifsListWidget::deleteUnusedInlineLayouts() { void GifsListWidget::deleteUnusedInlineLayouts() {
if (_inlineRows.isEmpty() || _section == Section::Gifs) { // delete all if (_rows.isEmpty() || _section == Section::Gifs) { // delete all
_inlineLayouts.clear(); _inlineLayouts.clear();
} else { } else {
for (auto i = _inlineLayouts.begin(); i != _inlineLayouts.cend();) { for (auto i = _inlineLayouts.begin(); i != _inlineLayouts.cend();) {
@ -436,7 +542,7 @@ void GifsListWidget::deleteUnusedInlineLayouts() {
} }
} }
GifsListWidget::InlineRow &GifsListWidget::layoutInlineRow(InlineRow &row, int32 sumWidth) { GifsListWidget::Row &GifsListWidget::layoutInlineRow(Row &row, int32 sumWidth) {
auto count = int(row.items.size()); auto count = int(row.items.size());
t_assert(count <= kInlineItemsMaxPerRow); t_assert(count <= kInlineItemsMaxPerRow);
@ -456,7 +562,7 @@ GifsListWidget::InlineRow &GifsListWidget::layoutInlineRow(InlineRow &row, int32
int index = indices[i]; int index = indices[i];
int w = sumWidth ? (row.items.at(index)->maxWidth() * availw / sumWidth) : row.items.at(index)->maxWidth(); int w = sumWidth ? (row.items.at(index)->maxWidth() * availw / sumWidth) : row.items.at(index)->maxWidth();
int actualw = qMax(w, int(st::inlineResultsMinWidth)); int actualw = qMax(w, int(st::inlineResultsMinWidth));
row.height = qMax(row.height, row.items.at(index)->resizeGetHeight(actualw)); row.height = qMax(row.height, row.items[index]->resizeGetHeight(actualw));
if (sumWidth) { if (sumWidth) {
availw -= actualw; availw -= actualw;
sumWidth -= row.items.at(index)->maxWidth(); sumWidth -= row.items.at(index)->maxWidth();
@ -470,14 +576,14 @@ GifsListWidget::InlineRow &GifsListWidget::layoutInlineRow(InlineRow &row, int32
} }
void GifsListWidget::preloadImages() { void GifsListWidget::preloadImages() {
for (auto row = 0, rows = _inlineRows.size(); row != rows; ++row) { for (auto row = 0, rows = _rows.size(); row != rows; ++row) {
for (auto col = 0, cols = _inlineRows[row].items.size(); col != cols; ++col) { for (auto col = 0, cols = _rows[row].items.size(); col != cols; ++col) {
_inlineRows[row].items[col]->preload(); _rows[row].items[col]->preload();
} }
} }
} }
void GifsListWidget::hideInlineRowsPanel() { void GifsListWidget::switchToSavedGifs() {
clearInlineRows(false); clearInlineRows(false);
_section = Section::Gifs; _section = Section::Gifs;
refreshSavedGifs(); refreshSavedGifs();
@ -485,66 +591,25 @@ void GifsListWidget::hideInlineRowsPanel() {
emit scrollUpdated(); emit scrollUpdated();
} }
void GifsListWidget::clearInlineRowsPanel() { int GifsListWidget::refreshInlineRows(const InlineCacheEntry *entry, bool resultsDeleted) {
clearInlineRows(false); if (!entry) {
}
void GifsListWidget::refreshSwitchPmButton(const InlineCacheEntry *entry) {
if (!entry || entry->switchPmText.isEmpty()) {
_switchPmButton.destroy();
_switchPmStartToken.clear();
} else {
if (!_switchPmButton) {
_switchPmButton.create(this, QString(), st::switchPmButton);
_switchPmButton->show();
_switchPmButton->setTextTransform(Ui::RoundButton::TextTransform::NoTransform);
connect(_switchPmButton, SIGNAL(clicked()), this, SLOT(onSwitchPm()));
}
_switchPmButton->setText(entry->switchPmText); // doesn't perform text.toUpper()
_switchPmStartToken = entry->switchPmStartToken;
auto buttonTop = st::stickerPanPadding;
_switchPmButton->move(st::inlineResultsLeft - st::buttonRadius, buttonTop);
}
update();
}
int GifsListWidget::refreshInlineRows(UserData *bot, const InlineCacheEntry *entry, bool resultsDeleted) {
_inlineBot = bot;
refreshSwitchPmButton(entry);
auto clearResults = [this, entry]() {
if (!entry) {
return true;
}
if (entry->results.empty() && entry->switchPmText.isEmpty()) {
if (!_inlineBot) {
return true;
}
}
return false;
};
auto clearResultsResult = clearResults();
if (clearResultsResult) {
if (resultsDeleted) { if (resultsDeleted) {
clearInlineRows(true); clearInlineRows(true);
deleteUnusedInlineLayouts(); deleteUnusedInlineLayouts();
} }
emit emptyInlineRows(); switchToSavedGifs();
return 0; return 0;
} }
clearSelection(); clearSelection();
t_assert(_inlineBot != 0);
_inlineBotTitle = lng_inline_bot_results(lt_inline_bot, _inlineBot->username.isEmpty() ? _inlineBot->name : ('@' + _inlineBot->username));
_section = Section::Inlines; _section = Section::Inlines;
auto count = int(entry->results.size()); auto count = int(entry->results.size());
auto from = validateExistingInlineRows(entry->results); auto from = validateExistingInlineRows(entry->results);
auto added = 0; auto added = 0;
if (count) { if (count) {
_inlineRows.reserve(count); _rows.reserve(count);
auto row = InlineRow(); auto row = Row();
row.items.reserve(kInlineItemsMaxPerRow); row.items.reserve(kInlineItemsMaxPerRow);
auto sumWidth = 0; auto sumWidth = 0;
for (auto i = from; i != count; ++i) { for (auto i = from; i != count; ++i) {
@ -568,52 +633,52 @@ int GifsListWidget::refreshInlineRows(UserData *bot, const InlineCacheEntry *ent
int GifsListWidget::validateExistingInlineRows(const InlineResults &results) { int GifsListWidget::validateExistingInlineRows(const InlineResults &results) {
int count = results.size(), until = 0, untilrow = 0, untilcol = 0; int count = results.size(), until = 0, untilrow = 0, untilcol = 0;
for (; until < count;) { for (; until < count;) {
if (untilrow >= _inlineRows.size() || _inlineRows[untilrow].items[untilcol]->getResult() != results[until].get()) { if (untilrow >= _rows.size() || _rows[untilrow].items[untilcol]->getResult() != results[until].get()) {
break; break;
} }
++until; ++until;
if (++untilcol == _inlineRows[untilrow].items.size()) { if (++untilcol == _rows[untilrow].items.size()) {
++untilrow; ++untilrow;
untilcol = 0; untilcol = 0;
} }
} }
if (until == count) { // all items are layed out if (until == count) { // all items are layed out
if (untilrow == _inlineRows.size()) { // nothing changed if (untilrow == _rows.size()) { // nothing changed
return until; return until;
} }
for (int i = untilrow, l = _inlineRows.size(), skip = untilcol; i < l; ++i) { for (int i = untilrow, l = _rows.size(), skip = untilcol; i < l; ++i) {
for (int j = 0, s = _inlineRows[i].items.size(); j < s; ++j) { for (int j = 0, s = _rows[i].items.size(); j < s; ++j) {
if (skip) { if (skip) {
--skip; --skip;
} else { } else {
_inlineRows[i].items[j]->setPosition(-1); _rows[i].items[j]->setPosition(-1);
} }
} }
} }
if (!untilcol) { // all good rows are filled if (!untilcol) { // all good rows are filled
_inlineRows.resize(untilrow); _rows.resize(untilrow);
return until; return until;
} }
_inlineRows.resize(untilrow + 1); _rows.resize(untilrow + 1);
_inlineRows[untilrow].items.resize(untilcol); _rows[untilrow].items.resize(untilcol);
_inlineRows[untilrow] = layoutInlineRow(_inlineRows[untilrow]); _rows[untilrow] = layoutInlineRow(_rows[untilrow]);
return until; return until;
} }
if (untilrow && !untilcol) { // remove last row, maybe it is not full if (untilrow && !untilcol) { // remove last row, maybe it is not full
--untilrow; --untilrow;
untilcol = _inlineRows[untilrow].items.size(); untilcol = _rows[untilrow].items.size();
} }
until -= untilcol; until -= untilcol;
for (int i = untilrow, l = _inlineRows.size(); i < l; ++i) { for (int i = untilrow, l = _rows.size(); i < l; ++i) {
for (int j = 0, s = _inlineRows[i].items.size(); j < s; ++j) { for (int j = 0, s = _rows[i].items.size(); j < s; ++j) {
_inlineRows[i].items[j]->setPosition(-1); _rows[i].items[j]->setPosition(-1);
} }
} }
_inlineRows.resize(untilrow); _rows.resize(untilrow);
if (_inlineRows.isEmpty()) { if (_rows.isEmpty()) {
_inlineWithThumb = false; _inlineWithThumb = false;
for (int i = until; i < count; ++i) { for (int i = until; i < count; ++i) {
if (results.at(i)->hasThumbDisplay()) { if (results.at(i)->hasThumbDisplay()) {
@ -625,20 +690,20 @@ int GifsListWidget::validateExistingInlineRows(const InlineResults &results) {
return until; return until;
} }
void GifsListWidget::inlineItemLayoutChanged(const InlineItem *layout) { void GifsListWidget::inlineItemLayoutChanged(const InlineBots::Layout::ItemBase *layout) {
if (_selected < 0 || !isVisible()) { if (_selected < 0 || !isVisible()) {
return; return;
} }
int row = _selected / MatrixRowShift, col = _selected % MatrixRowShift; int row = _selected / MatrixRowShift, col = _selected % MatrixRowShift;
if (row < _inlineRows.size() && col < _inlineRows.at(row).items.size()) { if (row < _rows.size() && col < _rows[row].items.size()) {
if (layout == _inlineRows.at(row).items.at(col)) { if (layout == _rows[row].items[col]) {
updateSelected(); updateSelected();
} }
} }
} }
void GifsListWidget::inlineItemRepaint(const InlineItem *layout) { void GifsListWidget::inlineItemRepaint(const InlineBots::Layout::ItemBase *layout) {
auto ms = getms(); auto ms = getms();
if (_lastScrolled + 100 <= ms) { if (_lastScrolled + 100 <= ms) {
update(); update();
@ -647,7 +712,7 @@ void GifsListWidget::inlineItemRepaint(const InlineItem *layout) {
} }
} }
bool GifsListWidget::inlineItemVisible(const InlineItem *layout) { bool GifsListWidget::inlineItemVisible(const InlineBots::Layout::ItemBase *layout) {
auto position = layout->position(); auto position = layout->position();
if (position < 0 || !isVisible()) { if (position < 0 || !isVisible()) {
return false; return false;
@ -655,15 +720,113 @@ bool GifsListWidget::inlineItemVisible(const InlineItem *layout) {
auto row = position / MatrixRowShift; auto row = position / MatrixRowShift;
auto col = position % MatrixRowShift; auto col = position % MatrixRowShift;
t_assert((row < _inlineRows.size()) && (col < _inlineRows[row].items.size())); t_assert((row < _rows.size()) && (col < _rows[row].items.size()));
auto &inlineItems = _inlineRows[row].items; auto &inlineItems = _rows[row].items;
auto top = 0; auto top = 0;
for (auto i = 0; i != row; ++i) { for (auto i = 0; i != row; ++i) {
top += _inlineRows[i].height; top += _rows[i].height;
} }
return (top < getVisibleBottom()) && (top + _inlineRows[row].items[col]->height() > getVisibleTop()); return (top < getVisibleBottom()) && (top + _rows[row].items[col]->height() > getVisibleTop());
}
void GifsListWidget::afterShown() {
if (_footer) {
_footer->stealFocus();
}
}
void GifsListWidget::beforeHiding() {
if (_footer) {
_footer->returnFocus();
}
}
bool GifsListWidget::refreshInlineRows(int32 *added) {
auto it = _inlineCache.find(_inlineQuery);
const InlineCacheEntry *entry = nullptr;
if (it != _inlineCache.cend()) {
if (!it->second->results.empty()) {
entry = it->second.get();
}
_inlineNextOffset = it->second->nextOffset;
}
auto result = refreshInlineRows(entry, false);
if (added) *added = result;
return (entry != nullptr);
}
int32 GifsListWidget::showInlineRows(bool newResults) {
auto added = 0;
auto clear = !refreshInlineRows(&added);
if (newResults) {
scrollToY(0);
}
return added;
}
void GifsListWidget::searchForGifs(const QString &query) {
if (query.isEmpty()) {
cancelGifsSearch();
return;
}
if (_inlineQuery != query) {
if (_inlineRequestId) {
request(_inlineRequestId).cancel();
_inlineRequestId = 0;
}
if (_inlineCache.find(query) != _inlineCache.cend()) {
_inlineRequestTimer.stop();
_inlineQuery = _inlineNextQuery = query;
showInlineRows(true);
} else {
_inlineNextQuery = query;
_inlineRequestTimer.start(kSearchRequestDelay);
}
}
if (!_searchBot && !_searchBotRequestId) {
_searchBotRequestId = request(MTPcontacts_ResolveUsername(MTP_string(kSearchBotUsername))).done([this](const MTPcontacts_ResolvedPeer &result) {
Expects(result.type() == mtpc_contacts_resolvedPeer);
auto &data = result.c_contacts_resolvedPeer();
App::feedUsers(data.vusers);
App::feedChats(data.vchats);
if (auto peer = App::peerLoaded(peerFromMTP(data.vpeer))) {
if (auto user = peer->asUser()) {
_searchBot = user;
}
}
}).send();
}
}
void GifsListWidget::sendInlineRequest() {
if (_inlineRequestId || !_inlineQueryPeer || _inlineNextQuery.isEmpty()) {
return;
}
if (!_searchBot) {
// Wait for the bot being resolved.
_inlineRequestTimer.start(kSearchRequestDelay);
return;
}
_inlineRequestTimer.stop();
_inlineQuery = _inlineNextQuery;
auto nextOffset = QString();
auto it = _inlineCache.find(_inlineQuery);
if (it != _inlineCache.cend()) {
nextOffset = it->second->nextOffset;
if (nextOffset.isEmpty()) return;
}
_inlineRequestId = request(MTPmessages_GetInlineBotResults(MTP_flags(0), _searchBot->inputUser, _inlineQueryPeer->input, MTPInputGeoPoint(), MTP_string(_inlineQuery), MTP_string(nextOffset))).done([this](const MTPmessages_BotResults &result, mtpRequestId requestId) {
inlineResultsDone(result);
}).fail([this](const RPCError &error) {
// show error?
_inlineRequestId = 0;
}).handleAllErrors().send();
} }
void GifsListWidget::refreshRecent() { void GifsListWidget::refreshRecent() {
@ -682,32 +845,29 @@ void GifsListWidget::updateSelected() {
int sx = (rtl() ? width() - p.x() : p.x()) - (st::inlineResultsLeft - st::buttonRadius); int sx = (rtl() ? width() - p.x() : p.x()) - (st::inlineResultsLeft - st::buttonRadius);
int sy = p.y() - st::stickerPanPadding; int sy = p.y() - st::stickerPanPadding;
if (_switchPmButton) {
sy -= _switchPmButton->height() + st::inlineResultsSkip;
}
int row = -1, col = -1, sel = -1; int row = -1, col = -1, sel = -1;
ClickHandlerPtr lnk; ClickHandlerPtr lnk;
ClickHandlerHost *lnkhost = nullptr; ClickHandlerHost *lnkhost = nullptr;
HistoryCursorState cursor = HistoryDefaultCursorState; HistoryCursorState cursor = HistoryDefaultCursorState;
if (sy >= 0) { if (sy >= 0) {
row = 0; row = 0;
for (int rows = _inlineRows.size(); row < rows; ++row) { for (int rows = _rows.size(); row < rows; ++row) {
if (sy < _inlineRows.at(row).height) { if (sy < _rows[row].height) {
break; break;
} }
sy -= _inlineRows.at(row).height; sy -= _rows[row].height;
} }
} }
if (sx >= 0 && row >= 0 && row < _inlineRows.size()) { if (sx >= 0 && row >= 0 && row < _rows.size()) {
auto &inlineItems = _inlineRows[row].items; auto &inlineItems = _rows[row].items;
col = 0; col = 0;
for (int cols = inlineItems.size(); col < cols; ++col) { for (int cols = inlineItems.size(); col < cols; ++col) {
int width = inlineItems.at(col)->width(); int width = inlineItems[col]->width();
if (sx < width) { if (sx < width) {
break; break;
} }
sx -= width; sx -= width;
if (inlineItems.at(col)->hasRightSkip()) { if (inlineItems[col]->hasRightSkip()) {
sx -= st::inlineResultsSkip; sx -= st::inlineResultsSkip;
} }
} }
@ -725,18 +885,18 @@ void GifsListWidget::updateSelected() {
int scol = (_selected >= 0) ? (_selected % MatrixRowShift) : -1; int scol = (_selected >= 0) ? (_selected % MatrixRowShift) : -1;
if (_selected != sel) { if (_selected != sel) {
if (srow >= 0 && scol >= 0) { if (srow >= 0 && scol >= 0) {
t_assert(srow >= 0 && srow < _inlineRows.size() && scol >= 0 && scol < _inlineRows.at(srow).items.size()); t_assert(srow >= 0 && srow < _rows.size() && scol >= 0 && scol < _rows[srow].items.size());
_inlineRows[srow].items[scol]->update(); _rows[srow].items[scol]->update();
} }
_selected = sel; _selected = sel;
if (row >= 0 && col >= 0) { if (row >= 0 && col >= 0) {
t_assert(row >= 0 && row < _inlineRows.size() && col >= 0 && col < _inlineRows.at(row).items.size()); t_assert(row >= 0 && row < _rows.size() && col >= 0 && col < _rows[row].items.size());
_inlineRows[row].items[col]->update(); _rows[row].items[col]->update();
} }
if (_previewShown && _selected >= 0 && _pressed != _selected) { if (_previewShown && _selected >= 0 && _pressed != _selected) {
_pressed = _selected; _pressed = _selected;
if (row >= 0 && col >= 0) { if (row >= 0 && col >= 0) {
auto layout = _inlineRows.at(row).items.at(col); auto layout = _rows[row].items[col];
if (auto previewDocument = layout->getPreviewDocument()) { if (auto previewDocument = layout->getPreviewDocument()) {
Ui::showMediaPreview(previewDocument); Ui::showMediaPreview(previewDocument);
} else if (auto previewPhoto = layout->getPreviewPhoto()) { } else if (auto previewPhoto = layout->getPreviewPhoto()) {
@ -753,8 +913,8 @@ void GifsListWidget::updateSelected() {
void GifsListWidget::onPreview() { void GifsListWidget::onPreview() {
if (_pressed < 0) return; if (_pressed < 0) return;
int row = _pressed / MatrixRowShift, col = _pressed % MatrixRowShift; int row = _pressed / MatrixRowShift, col = _pressed % MatrixRowShift;
if (row < _inlineRows.size() && col < _inlineRows.at(row).items.size()) { if (row < _rows.size() && col < _rows[row].items.size()) {
auto layout = _inlineRows.at(row).items.at(col); auto layout = _rows[row].items[col];
if (auto previewDocument = layout->getPreviewDocument()) { if (auto previewDocument = layout->getPreviewDocument()) {
Ui::showMediaPreview(previewDocument); Ui::showMediaPreview(previewDocument);
_previewShown = true; _previewShown = true;
@ -774,11 +934,4 @@ void GifsListWidget::onUpdateInlineItems() {
} }
} }
void GifsListWidget::onSwitchPm() {
if (_inlineBot && _inlineBot->botInfo) {
_inlineBot->botInfo->startToken = _switchPmStartToken;
Ui::showPeerHistory(_inlineBot, ShowAndStartBotMsgId);
}
}
} // namespace ChatHelpers } // namespace ChatHelpers

View File

@ -21,6 +21,14 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#pragma once #pragma once
#include "stickers/emoji_panel.h" #include "stickers/emoji_panel.h"
#include "inline_bots/inline_bot_layout_item.h"
namespace InlineBots {
namespace Layout {
class ItemBase;
} // namespace Layout
class Result;
} // namespace InlineBots
namespace Ui { namespace Ui {
class RoundButton; class RoundButton;
@ -28,7 +36,7 @@ class RoundButton;
namespace ChatHelpers { namespace ChatHelpers {
class GifsListWidget : public EmojiPanel::Inner, public InlineBots::Layout::Context, private base::Subscriber { class GifsListWidget : public EmojiPanel::Inner, public InlineBots::Layout::Context, private base::Subscriber, private MTP::Sender {
Q_OBJECT Q_OBJECT
public: public:
@ -39,17 +47,20 @@ public:
void clearSelection() override; void clearSelection() override;
object_ptr<EmojiPanel::InnerFooter> createFooter() override; object_ptr<EmojiPanel::InnerFooter> createFooter() override;
void refreshSavedGifs();
int refreshInlineRows(UserData *bot, const InlineCacheEntry *results, bool resultsDeleted);
void inlineBotChanged();
void hideInlineRowsPanel();
void clearInlineRowsPanel();
void setVisibleTopBottom(int visibleTop, int visibleBottom) override; void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
void inlineItemLayoutChanged(const InlineItem *layout) override; void inlineItemLayoutChanged(const InlineBots::Layout::ItemBase *layout) override;
void inlineItemRepaint(const InlineItem *layout) override; void inlineItemRepaint(const InlineBots::Layout::ItemBase *layout) override;
bool inlineItemVisible(const InlineItem *layout) override; bool inlineItemVisible(const InlineBots::Layout::ItemBase *layout) override;
void afterShown() override;
void beforeHiding() override;
void setInlineQueryPeer(PeerData *peer) {
_inlineQueryPeer = peer;
}
void searchForGifs(const QString &query);
void sendInlineRequest();
~GifsListWidget(); ~GifsListWidget();
@ -70,12 +81,12 @@ protected:
private slots: private slots:
void onPreview(); void onPreview();
void onUpdateInlineItems(); void onUpdateInlineItems();
void onSwitchPm();
signals: signals:
void selected(DocumentData *sticker); void selected(DocumentData *sticker);
void selected(PhotoData *photo); void selected(PhotoData *photo);
void selected(InlineBots::Result *result, UserData *bot); void selected(InlineBots::Result *result, UserData *bot);
void cancelled();
void emptyInlineRows(); void emptyInlineRows();
void scrollUpdated(); void scrollUpdated();
@ -87,39 +98,50 @@ private:
}; };
class Footer; class Footer;
using InlineResult = InlineBots::Result;
using InlineResults = std::vector<std::unique_ptr<InlineResult>>;
using LayoutItem = InlineBots::Layout::ItemBase;
struct InlineCacheEntry {
QString nextOffset;
InlineResults results;
};
void cancelGifsSearch();
void switchToSavedGifs();
void refreshSavedGifs();
int refreshInlineRows(const InlineCacheEntry *results, bool resultsDeleted);
void checkLoadMore();
int32 showInlineRows(bool newResults);
bool refreshInlineRows(int32 *added = 0);
void inlineResultsDone(const MTPmessages_BotResults &result);
void updateSelected(); void updateSelected();
void paintInlineItems(Painter &p, QRect clip); void paintInlineItems(Painter &p, QRect clip);
void refreshSwitchPmButton(const InlineCacheEntry *entry);
Section _section = Section::Gifs; Section _section = Section::Gifs;
UserData *_inlineBot;
QString _inlineBotTitle;
TimeMs _lastScrolled = 0; TimeMs _lastScrolled = 0;
QTimer _updateInlineItems; QTimer _updateInlineItems;
bool _inlineWithThumb = false; bool _inlineWithThumb = false;
object_ptr<Ui::RoundButton> _switchPmButton = { nullptr }; struct Row {
QString _switchPmStartToken;
typedef QVector<InlineItem*> InlineItems;
struct InlineRow {
int height = 0; int height = 0;
InlineItems items; QVector<LayoutItem*> items;
}; };
typedef QVector<InlineRow> InlineRows; QVector<Row> _rows;
InlineRows _inlineRows;
void clearInlineRows(bool resultsDeleted); void clearInlineRows(bool resultsDeleted);
std::map<DocumentData*, std::unique_ptr<InlineItem>> _gifLayouts; std::map<DocumentData*, std::unique_ptr<LayoutItem>> _gifLayouts;
InlineItem *layoutPrepareSavedGif(DocumentData *doc, int32 position); LayoutItem *layoutPrepareSavedGif(DocumentData *doc, int32 position);
std::map<InlineResult*, std::unique_ptr<InlineItem>> _inlineLayouts; std::map<InlineResult*, std::unique_ptr<LayoutItem>> _inlineLayouts;
InlineItem *layoutPrepareInlineResult(InlineResult *result, int32 position); LayoutItem *layoutPrepareInlineResult(InlineResult *result, int32 position);
bool inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, InlineRow &row, int32 &sumWidth); bool inlineRowsAddItem(DocumentData *savedGif, InlineResult *result, Row &row, int32 &sumWidth);
bool inlineRowFinalize(InlineRow &row, int32 &sumWidth, bool force = false); bool inlineRowFinalize(Row &row, int32 &sumWidth, bool force = false);
InlineRow &layoutInlineRow(InlineRow &row, int32 sumWidth = 0); Row &layoutInlineRow(Row &row, int32 sumWidth = 0);
void deleteUnusedGifLayouts(); void deleteUnusedGifLayouts();
void deleteUnusedInlineLayouts(); void deleteUnusedInlineLayouts();
@ -136,6 +158,15 @@ private:
QTimer _previewTimer; QTimer _previewTimer;
bool _previewShown = false; bool _previewShown = false;
std::map<QString, std::unique_ptr<InlineCacheEntry>> _inlineCache;
QTimer _inlineRequestTimer;
UserData *_searchBot = nullptr;
mtpRequestId _searchBotRequestId = 0;
PeerData *_inlineQueryPeer = nullptr;
QString _inlineQuery, _inlineNextQuery, _inlineNextOffset;
mtpRequestId _inlineRequestId = 0;
}; };
} // namespace ChatHelpers } // namespace ChatHelpers

View File

@ -211,3 +211,10 @@ inlineBotsScroll: ScrollArea(defaultSolidScroll) {
deltat: stickerPanPadding; deltat: stickerPanPadding;
deltab: stickerPanPadding; deltab: stickerPanPadding;
} }
gifsSearchField: contactsSearchField;
gifsSearchFieldPosition: point(42px, 7px);
gifsSearchCancel: contactsSearchCancel;
gifsSearchCancelPosition: point(1px, 1px);
gifsSearchIcon: boxFieldSearchIcon;
gifsSearchIconPosition: point(6px, 7px);

View File

@ -631,28 +631,14 @@ CrossButton::CrossButton(QWidget *parent, const style::CrossButton &st) : Ripple
hide(); hide();
} }
void CrossButton::showAnimated() { void CrossButton::toggleAnimated(bool visible) {
startAnimation(true); if (_shown == visible) {
} return;
}
void CrossButton::showFast() { _shown = visible;
showAnimated(); if (isHidden()) {
_a_show.finish(); show();
} }
void CrossButton::hideAnimated() {
startAnimation(false);
}
void CrossButton::hideFast() {
hideAnimated();
_a_show.finish();
}
void CrossButton::startAnimation(bool shown) {
if (_shown == shown) return;
_shown = shown;
if (isHidden()) show();
_a_show.start([this] { animationCallback(); }, _shown ? 0. : 1., _shown ? 1. : 0., _st.duration); _a_show.start([this] { animationCallback(); }, _shown ? 0. : 1., _shown ? 1. : 0., _st.duration);
} }

View File

@ -212,10 +212,23 @@ class CrossButton : public RippleButton {
public: public:
CrossButton(QWidget *parent, const style::CrossButton &st); CrossButton(QWidget *parent, const style::CrossButton &st);
void showAnimated(); void showAnimated() {
void showFast(); toggleAnimated(true);
void hideAnimated(); }
void hideFast(); void hideAnimated() {
toggleAnimated(false);
}
void toggleAnimated(bool visible);
void showFast() {
toggleFast(true);
}
void hideFast() {
toggleFast(false);
}
void toggleFast(bool visible) {
toggleAnimated(visible);
_a_show.finish();
}
bool isShown() const { bool isShown() const {
return _shown; return _shown;
@ -230,7 +243,6 @@ protected:
QPoint prepareRippleStartPosition() const override; QPoint prepareRippleStartPosition() const override;
private: private:
void startAnimation(bool shown);
void animationCallback(); void animationCallback();
const style::CrossButton &_st; const style::CrossButton &_st;