diff --git a/Telegram/SourceFiles/chat_helpers/chat_helpers.style b/Telegram/SourceFiles/chat_helpers/chat_helpers.style index d1bf2d91f..aca14e786 100644 --- a/Telegram/SourceFiles/chat_helpers/chat_helpers.style +++ b/Telegram/SourceFiles/chat_helpers/chat_helpers.style @@ -238,3 +238,21 @@ emojiSuggestionsMenu: Menu(defaultMenu) { itemPadding: margins(48px, 8px, 17px, 7px); widthMax: 512px; } + +mentionHeight: 40px; +mentionScroll: ScrollArea(defaultScrollArea) { + topsh: 0px; + bottomsh: 0px; +} +mentionPadding: margins(8px, 5px, 8px, 5px); +mentionTop: 11px; +mentionFont: linkFont; +mentionNameFg: windowFg; +mentionNameFgOver: windowFgOver; +mentionPhotoSize: msgPhotoSize; +mentionBg: windowBg; +mentionBgOver: windowBgOver; +mentionFg: windowSubTextFg; +mentionFgOver: windowSubTextFgOver; +mentionFgActive: windowActiveTextFg; +mentionFgOverActive: windowActiveTextFg; diff --git a/Telegram/SourceFiles/dialogswidget.cpp b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp similarity index 66% rename from Telegram/SourceFiles/dialogswidget.cpp rename to Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp index 9efff93c8..887688d40 100644 --- a/Telegram/SourceFiles/dialogswidget.cpp +++ b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.cpp @@ -18,36 +18,24 @@ to link the code of portions of this program with the OpenSSL library. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ -#include "dialogswidget.h" +#include "dialogs/dialogs_inner_widget.h" #include "dialogs/dialogs_indexed_list.h" #include "dialogs/dialogs_layout.h" #include "styles/style_dialogs.h" #include "styles/style_chat_helpers.h" -#include "styles/style_history.h" #include "ui/widgets/buttons.h" #include "ui/widgets/popup_menu.h" #include "data/data_drafts.h" #include "lang/lang_keys.h" -#include "application.h" #include "mainwindow.h" -#include "dialogswidget.h" #include "mainwidget.h" -#include "boxes/add_contact_box.h" -#include "boxes/contacts_box.h" -#include "boxes/confirm_box.h" -#include "boxes/about_box.h" #include "storage/localstorage.h" #include "apiwrap.h" -#include "ui/widgets/dropdown_menu.h" -#include "ui/widgets/input_fields.h" #include "window/themes/window_theme.h" -#include "autoupdater.h" #include "observer_peer.h" #include "auth_session.h" -#include "messenger.h" #include "window/notifications_manager.h" -#include "ui/effects/widget_fade_wrap.h" #include "window/window_controller.h" namespace { @@ -55,9 +43,6 @@ namespace { constexpr auto kHashtagResultsLimit = 5; constexpr auto kStartReorderThreshold = 30; -// Debug an assertion violation. -auto MustNotChangeDraggingIndex = false; - } // namespace struct DialogsInner::ImportantSwitch { @@ -98,7 +83,7 @@ DialogsInner::DialogsInner(QWidget *parent, gsl::not_null c } connect(main, SIGNAL(peerNameChanged(PeerData*, const PeerData::Names&, const PeerData::NameFirstChars&)), this, SLOT(onPeerNameChanged(PeerData*, const PeerData::Names&, const PeerData::NameFirstChars&))); connect(main, SIGNAL(peerPhotoChanged(PeerData*)), this, SLOT(onPeerPhotoChanged(PeerData*))); - connect(main, SIGNAL(dialogRowReplaced(Dialogs::Row*,Dialogs::Row*)), this, SLOT(onDialogRowReplaced(Dialogs::Row*,Dialogs::Row*))); + connect(main, SIGNAL(dialogRowReplaced(Dialogs::Row*, Dialogs::Row*)), this, SLOT(onDialogRowReplaced(Dialogs::Row*, Dialogs::Row*))); connect(_addContactLnk, SIGNAL(clicked()), App::wnd(), SLOT(onShowAddContact())); connect(_cancelSearchInPeer, SIGNAL(clicked()), this, SIGNAL(cancelSearchInPeer())); _cancelSearchInPeer->hide(); @@ -672,11 +657,6 @@ void DialogsInner::finishReorderPinned() { _dragging = nullptr; } - // Debug an assertion violation. - if (MustNotChangeDraggingIndex) { - Unexpected("Must not change draggingIndex while reordering pinned chats."); - } - _draggingIndex = -1; if (!_a_pinnedShifting.animating()) { _pinnedRows.clear(); @@ -706,11 +686,6 @@ int DialogsInner::updateReorderIndexGetCount() { return 0; } - // Debug an assertion violation. - if (MustNotChangeDraggingIndex) { - Unexpected("Must not change draggingIndex while reordering pinned chats."); - } - _draggingIndex = index; _aboveIndex = _draggingIndex; while (count > _pinnedRows.size()) { @@ -736,76 +711,23 @@ bool DialogsInner::updateReorderPinned(QPoint localPosition) { if (_dragStart.y() > localPosition.y() && _draggingIndex > 0) { shift = -floorclamp(_dragStart.y() - localPosition.y() + (rowHeight / 2), rowHeight, 0, _draggingIndex); - // Debug an assertion violation. - if (shift < 0) { - auto index = 0; - for_const (auto row, *shownDialogs()) { - if (++index >= _draggingIndex + shift) { - t_assert(row->history()->isPinnedDialog()); - if (index >= _draggingIndex) { - break; - } - } - } - } - MustNotChangeDraggingIndex = true; - for (auto from = _draggingIndex, to = _draggingIndex + shift; from > to; --from) { shownDialogs()->movePinned(_dragging, -1); std::swap(_pinnedRows[from], _pinnedRows[from - 1]); _pinnedRows[from].yadd = anim::value(_pinnedRows[from].yadd.current() - rowHeight, 0); _pinnedRows[from].animStartTime = ms; } - - // Debug an assertion violation. - MustNotChangeDraggingIndex = false; } else if (_dragStart.y() < localPosition.y() && _draggingIndex + 1 < pinnedCount) { shift = floorclamp(localPosition.y() - _dragStart.y() + (rowHeight / 2), rowHeight, 0, pinnedCount - _draggingIndex - 1); - // Debug an assertion violation. - MustNotChangeDraggingIndex = true; - if (shift > 0) { - auto index = 0; - for_const (auto row, *shownDialogs()) { - if (++index >= _draggingIndex) { - t_assert(row->history()->isPinnedDialog()); - if (index >= _draggingIndex + shift) { - break; - } - } - } - } - for (auto from = _draggingIndex, to = _draggingIndex + shift; from < to; ++from) { - - // Debug an assertion violation. - { - auto swapPinnedIndexWith = shownDialogs()->find(_dragging); - t_assert(swapPinnedIndexWith != shownDialogs()->cend()); - auto prevPos = (*swapPinnedIndexWith)->pos(); - ++swapPinnedIndexWith; - if (!(*swapPinnedIndexWith)->history()->isPinnedDialog()) { - SignalHandlers::setCrashAnnotation("DebugInfoBefore", QString("from: %1, to: %2, current: %3, prevPos: %4, nowPos: %5").arg(_draggingIndex).arg(_draggingIndex + shift).arg(from).arg(prevPos).arg((*swapPinnedIndexWith)->pos())); - } else { - SignalHandlers::setCrashAnnotation("DebugInfoBefore", QString()); - } - } - shownDialogs()->movePinned(_dragging, 1); std::swap(_pinnedRows[from], _pinnedRows[from + 1]); _pinnedRows[from].yadd = anim::value(_pinnedRows[from].yadd.current() + rowHeight, 0); _pinnedRows[from].animStartTime = ms; } - - // Debug an assertion violation. - MustNotChangeDraggingIndex = false; } if (shift) { - // Debug an assertion violation. - if (MustNotChangeDraggingIndex) { - Unexpected("Must not change draggingIndex while reordering pinned chats."); - } - _draggingIndex += shift; _aboveIndex = _draggingIndex; _dragStart.setY(_dragStart.y() + shift * rowHeight); @@ -2280,1080 +2202,3 @@ MsgId DialogsInner::lastSearchMigratedId() const { return _lastSearchMigratedId; } -class DialogsWidget::UpdateButton : public Ui::RippleButton { -public: - UpdateButton(QWidget *parent); - -protected: - void paintEvent(QPaintEvent *e) override; - - void onStateChanged(State was, StateChangeSource source) override; - -private: - QString _text; - const style::FlatButton &_st; - -}; - -DialogsWidget::UpdateButton::UpdateButton(QWidget *parent) : RippleButton(parent, st::dialogsUpdateButton.ripple) -, _text(lang(lng_update_telegram).toUpper()) -, _st(st::dialogsUpdateButton) { - resize(st::dialogsWidthMin, _st.height); -} - -void DialogsWidget::UpdateButton::onStateChanged(State was, StateChangeSource source) { - RippleButton::onStateChanged(was, source); - update(); -} - -void DialogsWidget::UpdateButton::paintEvent(QPaintEvent *e) { - QPainter p(this); - - QRect r(0, height() - _st.height, width(), _st.height); - p.fillRect(r, isOver() ? _st.overBgColor : _st.bgColor); - - paintRipple(p, 0, 0, getms()); - - p.setFont(isOver() ? _st.overFont : _st.font); - p.setRenderHint(QPainter::TextAntialiasing); - p.setPen(isOver() ? _st.overColor : _st.color); - - if (width() >= st::dialogsWidthMin) { - r.setTop(_st.textTop); - p.drawText(r, _text, style::al_top); - } else { - (isOver() ? st::dialogsInstallUpdateOver : st::dialogsInstallUpdate).paintInCenter(p, r); - } -} - -DialogsWidget::DialogsWidget(QWidget *parent, gsl::not_null controller) : Window::AbstractSectionWidget(parent, controller) -, _mainMenuToggle(this, st::dialogsMenuToggle) -, _filter(this, st::dialogsFilter, langFactory(lng_dlg_filter)) -, _searchFromUser(this, object_ptr(this, st::dialogsSearchFrom)) -, _jumpToDate(this, object_ptr(this, st::dialogsCalendar)) -, _cancelSearch(this, st::dialogsCancelSearch) -, _lockUnlock(this, st::dialogsLock) -, _scroll(this, st::dialogsScroll) { - _inner = _scroll->setOwnedWidget(object_ptr(this, controller, parent)); - connect(_inner, SIGNAL(draggingScrollDelta(int)), this, SLOT(onDraggingScrollDelta(int))); - connect(_inner, SIGNAL(mustScrollTo(int,int)), _scroll, SLOT(scrollToY(int,int))); - connect(_inner, SIGNAL(dialogMoved(int,int)), this, SLOT(onDialogMoved(int,int))); - connect(_inner, SIGNAL(searchMessages()), this, SLOT(onNeedSearchMessages())); - connect(_inner, SIGNAL(searchResultChosen()), this, SLOT(onCancel())); - connect(_inner, SIGNAL(completeHashtag(QString)), this, SLOT(onCompleteHashtag(QString))); - connect(_inner, SIGNAL(refreshHashtags()), this, SLOT(onFilterCursorMoved())); - connect(_inner, SIGNAL(cancelSearchInPeer()), this, SLOT(onCancelSearchInPeer())); - connect(_scroll, SIGNAL(geometryChanged()), _inner, SLOT(onParentGeometryChanged())); - connect(_scroll, SIGNAL(scrolled()), this, SLOT(onListScroll())); - connect(_filter, SIGNAL(cancelled()), this, SLOT(onCancel())); - connect(_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); - connect(_filter, SIGNAL(cursorPositionChanged(int,int)), this, SLOT(onFilterCursorMoved(int,int))); - -#ifndef TDESKTOP_DISABLE_AUTOUPDATE - Sandbox::connect(SIGNAL(updateLatest()), this, SLOT(onCheckUpdateStatus())); - Sandbox::connect(SIGNAL(updateFailed()), this, SLOT(onCheckUpdateStatus())); - Sandbox::connect(SIGNAL(updateReady()), this, SLOT(onCheckUpdateStatus())); - onCheckUpdateStatus(); -#endif // !TDESKTOP_DISABLE_AUTOUPDATE - - subscribe(Adaptive::Changed(), [this] { updateForwardBar(); }); - - _cancelSearch->setClickedCallback([this] { onCancelSearch(); }); - _jumpToDate->entity()->setClickedCallback([this] { if (_searchInPeer) this->controller()->showJumpToDate(_searchInPeer, QDate()); }); - _searchFromUser->entity()->setClickedCallback([this] { if (_searchInPeer->isChat() || _searchInPeer->isMegagroup()) showSearchFrom(); }); - _lockUnlock->setVisible(Global::LocalPasscode()); - subscribe(Global::RefLocalPasscodeChanged(), [this] { updateLockUnlockVisibility(); }); - _lockUnlock->setClickedCallback([this] { - _lockUnlock->setIconOverride(&st::dialogsUnlockIcon, &st::dialogsUnlockIconOver); - Messenger::Instance().setupPasscode(); - _lockUnlock->setIconOverride(nullptr); - }); - _mainMenuToggle->setClickedCallback([this] { showMainMenu(); }); - - _chooseByDragTimer.setSingleShot(true); - connect(&_chooseByDragTimer, SIGNAL(timeout()), this, SLOT(onChooseByDrag())); - - setAcceptDrops(true); - - _searchTimer.setSingleShot(true); - connect(&_searchTimer, SIGNAL(timeout()), this, SLOT(onSearchMessages())); - - _inner->setLoadMoreCallback([this] { - if (_inner->state() == DialogsInner::SearchedState || (_inner->state() == DialogsInner::FilteredState && _searchInMigrated && _searchFull && !_searchFullMigrated)) { - onSearchMore(); - } else { - loadDialogs(); - } - }); - - _filter->setFocusPolicy(Qt::StrongFocus); - _filter->customUpDown(true); - - updateJumpToDateVisibility(true); -} - -#ifndef TDESKTOP_DISABLE_AUTOUPDATE -void DialogsWidget::onCheckUpdateStatus() { - if (Sandbox::updatingState() == Application::UpdatingReady) { - if (_updateTelegram) return; - _updateTelegram.create(this); - _updateTelegram->show(); - _updateTelegram->setClickedCallback([] { - checkReadyUpdate(); - App::restart(); - }); - } else { - if (!_updateTelegram) return; - _updateTelegram.destroy(); - } - updateControlsGeometry(); -} -#endif // TDESKTOP_DISABLE_AUTOUPDATE - -void DialogsWidget::activate() { - _filter->setFocus(); - _inner->activate(); -} - -void DialogsWidget::createDialog(History *history) { - auto creating = !history->inChatList(Dialogs::Mode::All); - _inner->createDialog(history); - if (creating && history->peer->migrateFrom()) { - if (auto migrated = App::historyLoaded(history->peer->migrateFrom()->id)) { - if (migrated->inChatList(Dialogs::Mode::All)) { - removeDialog(migrated); - } - } - } -} - -void DialogsWidget::dlgUpdated(Dialogs::Mode list, Dialogs::Row *row) { - _inner->dlgUpdated(list, row); -} - -void DialogsWidget::dlgUpdated(PeerData *peer, MsgId msgId) { - _inner->dlgUpdated(peer, msgId); -} - -void DialogsWidget::dialogsToUp() { - if (_filter->getLastText().trimmed().isEmpty()) { - _scroll->scrollToY(0); - } -} - -void DialogsWidget::startWidthAnimation() { - if (!_widthAnimationCache.isNull()) { - return; - } - auto scrollGeometry = _scroll->geometry(); - auto grabGeometry = QRect(scrollGeometry.x(), scrollGeometry.y(), st::dialogsWidthMin, scrollGeometry.height()); - _scroll->setGeometry(grabGeometry); - myEnsureResized(_scroll); - auto image = QImage(grabGeometry.size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); - image.setDevicePixelRatio(cRetinaFactor()); - image.fill(Qt::transparent); - _scroll->render(&image, QPoint(0, 0), QRect(QPoint(0, 0), grabGeometry.size()), QWidget::DrawChildren | QWidget::IgnoreMask); - _widthAnimationCache = App::pixmapFromImageInPlace(std::move(image)); - _scroll->setGeometry(scrollGeometry); - _scroll->hide(); -} - -void DialogsWidget::stopWidthAnimation() { - _widthAnimationCache = QPixmap(); - if (!_a_show.animating()) { - _scroll->show(); - } - update(); -} - -void DialogsWidget::showFast() { - show(); - updateForwardBar(); -} - -void DialogsWidget::showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms) { - _showDirection = direction; - - _a_show.finish(); - - _cacheUnder = params.oldContentCache; - show(); - updateForwardBar(); - _cacheOver = App::main()->grabForShowAnimation(params); - - _scroll->hide(); - _mainMenuToggle->hide(); - if (_forwardCancel) _forwardCancel->hide(); - _filter->hide(); - _cancelSearch->hideFast(); - _jumpToDate->hideFast(); - _searchFromUser->hideFast(); - _lockUnlock->hide(); - - int delta = st::slideShift; - if (_showDirection == Window::SlideDirection::FromLeft) { - std::swap(_cacheUnder, _cacheOver); - } - _a_show.start([this] { animationCallback(); }, 0., 1., st::slideDuration, Window::SlideAnimation::transition()); -} - -bool DialogsWidget::wheelEventFromFloatPlayer(QEvent *e, Window::Column myColumn, Window::Column playerColumn) { - return _scroll->viewportEvent(e); -} - -QRect DialogsWidget::rectForFloatPlayer(Window::Column myColumn, Window::Column playerColumn) { - return mapToGlobal(_scroll->geometry()); -} - -void DialogsWidget::animationCallback() { - update(); - if (!_a_show.animating()) { - _cacheUnder = _cacheOver = QPixmap(); - - _scroll->show(); - _mainMenuToggle->show(); - if (_forwardCancel) _forwardCancel->show(); - _filter->show(); - updateLockUnlockVisibility(); - updateJumpToDateVisibility(true); - - onFilterUpdate(); - if (App::wnd()) App::wnd()->setInnerFocus(); - } -} - -void DialogsWidget::onCancel() { - if (!onCancelSearch() || (!_searchInPeer && !App::main()->selectingPeer())) { - emit cancelled(); - } -} - -void DialogsWidget::notify_userIsContactChanged(UserData *user, bool fromThisApp) { - if (fromThisApp) { - _filter->setText(QString()); - _filter->updatePlaceholder(); - onFilterUpdate(); - } - _inner->notify_userIsContactChanged(user, fromThisApp); -} - -void DialogsWidget::notify_historyMuteUpdated(History *history) { - _inner->notify_historyMuteUpdated(history); -} - -void DialogsWidget::unreadCountsReceived(const QVector &dialogs) { -} - -void DialogsWidget::dialogsReceived(const MTPmessages_Dialogs &dialogs, mtpRequestId requestId) { - if (_dialogsRequestId != requestId) return; - - const QVector *dialogsList = 0; - const QVector *messagesList = 0; - switch (dialogs.type()) { - case mtpc_messages_dialogs: { - auto &data = dialogs.c_messages_dialogs(); - App::feedUsers(data.vusers); - App::feedChats(data.vchats); - messagesList = &data.vmessages.v; - dialogsList = &data.vdialogs.v; - _dialogsFull = true; - } break; - case mtpc_messages_dialogsSlice: { - auto &data = dialogs.c_messages_dialogsSlice(); - App::feedUsers(data.vusers); - App::feedChats(data.vchats); - messagesList = &data.vmessages.v; - dialogsList = &data.vdialogs.v; - } break; - } - - if (!AuthSession::Current().data().contactsLoaded().value() && !_contactsRequestId) { - _contactsRequestId = MTP::send(MTPcontacts_GetContacts(MTP_string("")), rpcDone(&DialogsWidget::contactsReceived), rpcFail(&DialogsWidget::contactsFailed)); - } - - if (dialogsList) { - TimeId lastDate = 0; - PeerId lastPeer = 0; - MsgId lastMsgId = 0; - for (int i = dialogsList->size(); i > 0;) { - auto &dialog = dialogsList->at(--i); - if (dialog.type() != mtpc_dialog) { - continue; - } - - auto &dialogData = dialog.c_dialog(); - if (auto peer = peerFromMTP(dialogData.vpeer)) { - auto history = App::history(peer); - history->setPinnedDialog(dialogData.is_pinned()); - - if (!lastDate) { - if (!lastPeer) lastPeer = peer; - if (auto msgId = dialogData.vtop_message.v) { - if (!lastMsgId) lastMsgId = msgId; - for (int j = messagesList->size(); j > 0;) { - auto &message = messagesList->at(--j); - if (idFromMessage(message) == msgId && peerFromMessage(message) == peer) { - if (auto date = dateFromMessage(message)) { - lastDate = date; - } - break; - } - } - } - } - } - } - if (lastDate) { - _dialogsOffsetDate = lastDate; - _dialogsOffsetId = lastMsgId; - _dialogsOffsetPeer = App::peer(lastPeer); - } else { - _dialogsFull = true; - } - - t_assert(messagesList != nullptr); - App::feedMsgs(*messagesList, NewMessageLast); - - unreadCountsReceived(*dialogsList); - _inner->dialogsReceived(*dialogsList); - onListScroll(); - } else { - _dialogsFull = true; - } - - _dialogsRequestId = 0; - loadDialogs(); - - AuthSession::Current().data().moreChatsLoaded().notify(); - if (_dialogsFull) { - AuthSession::Current().data().allChatsLoaded().set(true); - } -} - -void DialogsWidget::pinnedDialogsReceived(const MTPmessages_PeerDialogs &dialogs, mtpRequestId requestId) { - if (_pinnedDialogsRequestId != requestId) return; - - if (dialogs.type() == mtpc_messages_peerDialogs) { - App::histories().clearPinned(); - - auto &dialogsData = dialogs.c_messages_peerDialogs(); - App::feedUsers(dialogsData.vusers); - App::feedChats(dialogsData.vchats); - auto &list = dialogsData.vdialogs.v; - for (auto i = list.size(); i > 0;) { - auto &dialog = list[--i]; - if (dialog.type() != mtpc_dialog) { - continue; - } - - auto &dialogData = dialog.c_dialog(); - if (auto peer = peerFromMTP(dialogData.vpeer)) { - auto history = App::history(peer); - history->setPinnedDialog(dialogData.is_pinned()); - } - } - App::feedMsgs(dialogsData.vmessages, NewMessageLast); - unreadCountsReceived(list); - _inner->dialogsReceived(list); - onListScroll(); - } - - _pinnedDialogsRequestId = 0; - _pinnedDialogsReceived = true; - - AuthSession::Current().data().moreChatsLoaded().notify(); -} - -bool DialogsWidget::dialogsFailed(const RPCError &error, mtpRequestId requestId) { - if (MTP::isDefaultHandledError(error)) return false; - - LOG(("RPC Error: %1 %2: %3").arg(error.code()).arg(error.type()).arg(error.description())); - if (_dialogsRequestId == requestId) { - _dialogsRequestId = 0; - } else if (_pinnedDialogsRequestId == requestId) { - _pinnedDialogsRequestId = 0; - } - return true; -} - -void DialogsWidget::onDraggingScrollDelta(int delta) { - _draggingScrollDelta = _scroll ? delta : 0; - if (_draggingScrollDelta) { - if (!_draggingScrollTimer) { - _draggingScrollTimer.create(this); - _draggingScrollTimer->setSingleShot(false); - connect(_draggingScrollTimer, SIGNAL(timeout()), this, SLOT(onDraggingScrollTimer())); - } - _draggingScrollTimer->start(15); - } else { - _draggingScrollTimer.destroy(); - } -} - -void DialogsWidget::onDraggingScrollTimer() { - auto delta = (_draggingScrollDelta > 0) ? qMin(_draggingScrollDelta * 3 / 20 + 1, int32(MaxScrollSpeed)) : qMax(_draggingScrollDelta * 3 / 20 - 1, -int32(MaxScrollSpeed)); - _scroll->scrollToY(_scroll->scrollTop() + delta); -} - -bool DialogsWidget::onSearchMessages(bool searchCache) { - QString q = _filter->getLastText().trimmed(); - if (q.isEmpty()) { - MTP::cancel(base::take(_searchRequest)); - MTP::cancel(base::take(_peerSearchRequest)); - return true; - } - if (searchCache) { - SearchCache::const_iterator i = _searchCache.constFind(q); - if (i != _searchCache.cend()) { - _searchQuery = q; - _searchFull = _searchFullMigrated = false; - MTP::cancel(base::take(_searchRequest)); - searchReceived(_searchInPeer ? DialogsSearchPeerFromStart : DialogsSearchFromStart, i.value(), 0); - return true; - } - } else if (_searchQuery != q) { - _searchQuery = q; - _searchFull = _searchFullMigrated = false; - MTP::cancel(base::take(_searchRequest)); - if (_searchInPeer) { - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInPeer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, DialogsSearchPeerFromStart), rpcFail(&DialogsWidget::searchFailed, DialogsSearchPeerFromStart)); - } else { - _searchRequest = MTP::send(MTPmessages_SearchGlobal(MTP_string(_searchQuery), MTP_int(0), MTP_inputPeerEmpty(), MTP_int(0), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, DialogsSearchFromStart), rpcFail(&DialogsWidget::searchFailed, DialogsSearchFromStart)); - } - _searchQueries.insert(_searchRequest, _searchQuery); - } - if (!_searchInPeer && q.size() >= MinUsernameLength) { - if (searchCache) { - auto i = _peerSearchCache.constFind(q); - if (i != _peerSearchCache.cend()) { - _peerSearchQuery = q; - _peerSearchRequest = 0; - peerSearchReceived(i.value(), 0); - return true; - } - } else if (_peerSearchQuery != q) { - _peerSearchQuery = q; - _peerSearchFull = false; - _peerSearchRequest = MTP::send(MTPcontacts_Search(MTP_string(_peerSearchQuery), MTP_int(SearchPeopleLimit)), rpcDone(&DialogsWidget::peerSearchReceived), rpcFail(&DialogsWidget::peopleFailed)); - _peerSearchQueries.insert(_peerSearchRequest, _peerSearchQuery); - } - } - return false; -} - -void DialogsWidget::onNeedSearchMessages() { - if (!onSearchMessages(true)) { - _searchTimer.start(AutoSearchTimeout); - } -} - -void DialogsWidget::onChooseByDrag() { - _inner->choosePeer(); -} - -void DialogsWidget::showMainMenu() { - App::wnd()->showMainMenu(); -} - -void DialogsWidget::searchMessages(const QString &query, PeerData *inPeer) { - if ((_filter->getLastText() != query) || (inPeer && inPeer != _searchInPeer && inPeer->migrateTo() != _searchInPeer)) { - if (inPeer) { - onCancelSearch(); - setSearchInPeer(inPeer); - } - _filter->setText(query); - _filter->updatePlaceholder(); - onFilterUpdate(true); - _searchTimer.stop(); - onSearchMessages(); - - _inner->saveRecentHashtags(query); - } -} - -void DialogsWidget::onSearchMore() { - if (!_searchRequest) { - if (!_searchFull) { - auto offsetDate = _inner->lastSearchDate(); - auto offsetPeer = _inner->lastSearchPeer(); - auto offsetId = _inner->lastSearchId(); - if (_searchInPeer) { - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInPeer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(offsetId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetId ? DialogsSearchPeerFromOffset : DialogsSearchPeerFromStart), rpcFail(&DialogsWidget::searchFailed, offsetId ? DialogsSearchPeerFromOffset : DialogsSearchPeerFromStart)); - } else { - _searchRequest = MTP::send(MTPmessages_SearchGlobal(MTP_string(_searchQuery), MTP_int(offsetDate), offsetPeer ? offsetPeer->input : MTP_inputPeerEmpty(), MTP_int(offsetId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetId ? DialogsSearchFromOffset : DialogsSearchFromStart), rpcFail(&DialogsWidget::searchFailed, offsetId ? DialogsSearchFromOffset : DialogsSearchFromStart)); - } - if (!offsetId) { - _searchQueries.insert(_searchRequest, _searchQuery); - } - } else if (_searchInMigrated && !_searchFullMigrated) { - auto offsetMigratedId = _inner->lastSearchMigratedId(); - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInMigrated->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(offsetMigratedId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetMigratedId ? DialogsSearchMigratedFromOffset : DialogsSearchMigratedFromStart), rpcFail(&DialogsWidget::searchFailed, offsetMigratedId ? DialogsSearchMigratedFromOffset : DialogsSearchMigratedFromStart)); - } - } -} - -void DialogsWidget::loadDialogs() { - if (_dialogsRequestId) return; - if (_dialogsFull) { - _inner->addAllSavedPeers(); - return; - } - - auto firstLoad = !_dialogsOffsetDate; - auto loadCount = firstLoad ? DialogsFirstLoad : DialogsPerPage; - auto flags = MTPmessages_GetDialogs::Flag::f_exclude_pinned; - _dialogsRequestId = MTP::send(MTPmessages_GetDialogs(MTP_flags(flags), MTP_int(_dialogsOffsetDate), MTP_int(_dialogsOffsetId), _dialogsOffsetPeer ? _dialogsOffsetPeer->input : MTP_inputPeerEmpty(), MTP_int(loadCount)), rpcDone(&DialogsWidget::dialogsReceived), rpcFail(&DialogsWidget::dialogsFailed)); - if (!_pinnedDialogsReceived) { - loadPinnedDialogs(); - } -} - -void DialogsWidget::loadPinnedDialogs() { - if (_pinnedDialogsRequestId) return; - - _pinnedDialogsReceived = false; - _pinnedDialogsRequestId = MTP::send(MTPmessages_GetPinnedDialogs(), rpcDone(&DialogsWidget::pinnedDialogsReceived), rpcFail(&DialogsWidget::dialogsFailed)); -} - -void DialogsWidget::contactsReceived(const MTPcontacts_Contacts &result) { - _contactsRequestId = 0; - if (result.type() == mtpc_contacts_contacts) { - auto &d = result.c_contacts_contacts(); - App::feedUsers(d.vusers); - _inner->contactsReceived(d.vcontacts.v); - } - AuthSession::Current().data().contactsLoaded().set(true); -} - -bool DialogsWidget::contactsFailed(const RPCError &error) { - if (MTP::isDefaultHandledError(error)) return false; - _contactsRequestId = 0; - return true; -} - -void DialogsWidget::searchReceived(DialogsSearchRequestType type, const MTPmessages_Messages &result, mtpRequestId req) { - if (_inner->state() == DialogsInner::FilteredState || _inner->state() == DialogsInner::SearchedState) { - if (type == DialogsSearchFromStart || type == DialogsSearchPeerFromStart) { - auto i = _searchQueries.find(req); - if (i != _searchQueries.cend()) { - _searchCache[i.value()] = result; - _searchQueries.erase(i); - } - } - } - - if (_searchRequest == req) { - switch (result.type()) { - case mtpc_messages_messages: { - auto &d = result.c_messages_messages(); - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - auto &msgs = d.vmessages.v; - if (!_inner->searchReceived(msgs, type, msgs.size())) { - if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { - _searchFullMigrated = true; - } else { - _searchFull = true; - } - } - } break; - - case mtpc_messages_messagesSlice: { - auto &d = result.c_messages_messagesSlice(); - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - auto &msgs = d.vmessages.v; - if (!_inner->searchReceived(msgs, type, d.vcount.v)) { - if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { - _searchFullMigrated = true; - } else { - _searchFull = true; - } - } - } break; - - case mtpc_messages_channelMessages: { - auto &d = result.c_messages_channelMessages(); - if (_searchInPeer && _searchInPeer->isChannel()) { - _searchInPeer->asChannel()->ptsReceived(d.vpts.v); - } else { - LOG(("API Error: received messages.channelMessages when no channel was passed! (DialogsWidget::searchReceived)")); - } - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - auto &msgs = d.vmessages.v; - if (!_inner->searchReceived(msgs, type, d.vcount.v)) { - if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { - _searchFullMigrated = true; - } else { - _searchFull = true; - } - } - } break; - } - - _searchRequest = 0; - onListScroll(); - update(); - } -} - -void DialogsWidget::peerSearchReceived(const MTPcontacts_Found &result, mtpRequestId req) { - auto q = _peerSearchQuery; - if (_inner->state() == DialogsInner::FilteredState || _inner->state() == DialogsInner::SearchedState) { - auto i = _peerSearchQueries.find(req); - if (i != _peerSearchQueries.cend()) { - q = i.value(); - _peerSearchCache[q] = result; - _peerSearchQueries.erase(i); - } - } - if (_peerSearchRequest == req) { - switch (result.type()) { - case mtpc_contacts_found: { - auto &d = result.c_contacts_found(); - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - _inner->peerSearchReceived(q, d.vresults.v); - } break; - } - - _peerSearchRequest = 0; - onListScroll(); - } -} - -bool DialogsWidget::searchFailed(DialogsSearchRequestType type, const RPCError &error, mtpRequestId req) { - if (MTP::isDefaultHandledError(error)) return false; - - if (_searchRequest == req) { - _searchRequest = 0; - if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { - _searchFullMigrated = true; - } else { - _searchFull = true; - } - } - return true; -} - -bool DialogsWidget::peopleFailed(const RPCError &error, mtpRequestId req) { - if (MTP::isDefaultHandledError(error)) return false; - - if (_peerSearchRequest == req) { - _peerSearchRequest = 0; - _peerSearchFull = true; - } - return true; -} - -void DialogsWidget::dragEnterEvent(QDragEnterEvent *e) { - if (App::main()->selectingPeer()) return; - - _dragInScroll = false; - _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-selected")); - if (!_dragForward) _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-pressed-link")); - if (!_dragForward) _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-pressed")); - if (_dragForward && Adaptive::OneColumn()) _dragForward = false; - if (_dragForward) { - e->setDropAction(Qt::CopyAction); - e->accept(); - updateDragInScroll(_scroll->geometry().contains(e->pos())); - } else if (App::main() && App::main()->getDragState(e->mimeData()) != DragStateNone) { - e->setDropAction(Qt::CopyAction); - e->accept(); - } - _chooseByDragTimer.stop(); -} - -void DialogsWidget::dragMoveEvent(QDragMoveEvent *e) { - if (_scroll->geometry().contains(e->pos())) { - if (_dragForward) { - updateDragInScroll(true); - } else { - _chooseByDragTimer.start(ChoosePeerByDragTimeout); - } - PeerData *p = _inner->updateFromParentDrag(mapToGlobal(e->pos())); - if (p) { - e->setDropAction(Qt::CopyAction); - } else { - e->setDropAction(Qt::IgnoreAction); - } - } else { - if (_dragForward) updateDragInScroll(false); - _inner->dragLeft(); - e->setDropAction(Qt::IgnoreAction); - } - e->accept(); -} - -void DialogsWidget::dragLeaveEvent(QDragLeaveEvent *e) { - if (_dragForward) { - updateDragInScroll(false); - } else { - _chooseByDragTimer.stop(); - } - _inner->dragLeft(); - e->accept(); -} - -void DialogsWidget::updateDragInScroll(bool inScroll) { - if (_dragInScroll != inScroll) { - _dragInScroll = inScroll; - if (_dragInScroll) { - App::main()->showForwardLayer(SelectedItemSet()); - } else { - App::main()->dialogsCancelled(); - } - } -} - -void DialogsWidget::dropEvent(QDropEvent *e) { - _chooseByDragTimer.stop(); - if (_scroll->geometry().contains(e->pos())) { - if (auto peer = _inner->updateFromParentDrag(mapToGlobal(e->pos()))) { - e->acceptProposedAction(); - App::main()->onFilesOrForwardDrop(peer->id, e->mimeData()); - } - } -} - -void DialogsWidget::onListScroll() { - auto scrollTop = _scroll->scrollTop(); - _inner->setVisibleTopBottom(scrollTop, scrollTop + _scroll->height()); -} - -void DialogsWidget::onFilterUpdate(bool force) { - if (_a_show.animating() && !force) return; - - auto filterText = _filter->getLastText(); - _inner->onFilterUpdate(filterText, force); - if (filterText.isEmpty()) { - _searchCache.clear(); - _searchQueries.clear(); - _searchQuery = QString(); - _cancelSearch->hideAnimated(); - } else { - _cancelSearch->showAnimated(); - } - updateJumpToDateVisibility(); - - if (filterText.size() < MinUsernameLength) { - _peerSearchCache.clear(); - _peerSearchQueries.clear(); - _peerSearchQuery = QString(); - } -} - -void DialogsWidget::searchInPeer(PeerData *peer) { - onCancelSearch(); - setSearchInPeer(peer); - onFilterUpdate(true); -} - -void DialogsWidget::setSearchInPeer(PeerData *peer) { - auto newSearchInPeer = peer ? (peer->migrateTo() ? peer->migrateTo() : peer) : nullptr; - _searchInMigrated = newSearchInPeer ? newSearchInPeer->migrateFrom() : nullptr; - if (newSearchInPeer != _searchInPeer) { - _searchInPeer = newSearchInPeer; - controller()->searchInPeerChanged().notify(_searchInPeer, true); - updateJumpToDateVisibility(); - } - _inner->searchInPeer(_searchInPeer); -} - -void DialogsWidget::showSearchFrom() { - -} - -void DialogsWidget::onFilterCursorMoved(int from, int to) { - if (to < 0) to = _filter->cursorPosition(); - QString t = _filter->getLastText(); - QStringRef r; - for (int start = to; start > 0;) { - --start; - if (t.size() <= start) break; - if (t.at(start) == '#') { - r = t.midRef(start, to - start); - break; - } - if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break; - } - _inner->onHashtagFilterUpdate(r); -} - -void DialogsWidget::onCompleteHashtag(QString tag) { - QString t = _filter->getLastText(), r; - int cur = _filter->cursorPosition(); - for (int start = cur; start > 0;) { - --start; - if (t.size() <= start) break; - if (t.at(start) == '#') { - if (cur == start + 1 || t.midRef(start + 1, cur - start - 1) == tag.midRef(0, cur - start - 1)) { - for (; cur < t.size() && cur - start - 1 < tag.size(); ++cur) { - if (t.at(cur) != tag.at(cur - start - 1)) break; - } - if (cur - start - 1 == tag.size() && cur < t.size() && t.at(cur) == ' ') ++cur; - r = t.mid(0, start + 1) + tag + ' ' + t.mid(cur); - _filter->setText(r); - _filter->setCursorPosition(start + 1 + tag.size() + 1); - onFilterUpdate(true); - return; - } - break; - } - if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break; - } - _filter->setText(t.mid(0, cur) + '#' + tag + ' ' + t.mid(cur)); - _filter->setCursorPosition(cur + 1 + tag.size() + 1); - onFilterUpdate(true); -} - -void DialogsWidget::resizeEvent(QResizeEvent *e) { - updateControlsGeometry(); -} - -void DialogsWidget::updateLockUnlockVisibility() { - if (!_a_show.animating()) { - _lockUnlock->setVisible(Global::LocalPasscode()); - } - updateControlsGeometry(); -} - -void DialogsWidget::updateJumpToDateVisibility(bool fast) { - if (_a_show.animating()) return; - - auto jumpToDateVisible = (_searchInPeer && _filter->getLastText().isEmpty()); - if (fast) { - _jumpToDate->toggleFast(jumpToDateVisible); - } else { - _jumpToDate->toggleAnimated(jumpToDateVisible); - } - - auto searchFromUserVisible = _searchInPeer && (_searchInPeer->isChat() || _searchInPeer->isMegagroup()); - if (fast) { - _searchFromUser->toggleFast(searchFromUserVisible); - } else { - _searchFromUser->toggleAnimated(searchFromUserVisible); - } -} - -void DialogsWidget::updateControlsGeometry() { - auto filterAreaTop = 0; - if (_forwardCancel) { - _forwardCancel->moveToLeft(0, filterAreaTop); - filterAreaTop += st::dialogsForwardHeight; - } - auto smallLayoutWidth = (st::dialogsPadding.x() + st::dialogsPhotoSize + st::dialogsPadding.x()); - auto smallLayoutRatio = (width() < st::dialogsWidthMin) ? (st::dialogsWidthMin - width()) / float64(st::dialogsWidthMin - smallLayoutWidth) : 0.; - auto filterLeft = st::dialogsFilterPadding.x() + _mainMenuToggle->width() + st::dialogsFilterPadding.x(); - auto filterRight = (Global::LocalPasscode() ? (st::dialogsFilterPadding.x() + _lockUnlock->width()) : st::dialogsFilterSkip) + st::dialogsFilterPadding.x(); - auto filterWidth = qMax(width(), st::dialogsWidthMin) - filterLeft - filterRight; - auto filterAreaHeight = st::dialogsFilterPadding.y() + _mainMenuToggle->height() + st::dialogsFilterPadding.y(); - auto filterTop = filterAreaTop + (filterAreaHeight - _filter->height()) / 2; - filterLeft = anim::interpolate(filterLeft, smallLayoutWidth, smallLayoutRatio); - _filter->setGeometryToLeft(filterLeft, filterTop, filterWidth, _filter->height()); - auto mainMenuLeft = anim::interpolate(st::dialogsFilterPadding.x(), (smallLayoutWidth - _mainMenuToggle->width()) / 2, smallLayoutRatio); - _mainMenuToggle->moveToLeft(mainMenuLeft, filterAreaTop + st::dialogsFilterPadding.y()); - _lockUnlock->moveToLeft(filterLeft + filterWidth + st::dialogsFilterPadding.x(), filterAreaTop + st::dialogsFilterPadding.y()); - _cancelSearch->moveToLeft(filterLeft + filterWidth - _cancelSearch->width(), _filter->y()); - _jumpToDate->moveToLeft(filterLeft + filterWidth - _jumpToDate->width(), _filter->y()); - _searchFromUser->moveToLeft(filterLeft + filterWidth - _jumpToDate->width() - _searchFromUser->width(), _filter->y()); - - auto scrollTop = filterAreaTop + filterAreaHeight; - auto addToScroll = App::main() ? App::main()->contentScrollAddToY() : 0; - auto newScrollTop = _scroll->scrollTop() + addToScroll; - auto scrollHeight = height() - scrollTop; - if (_updateTelegram) { - auto updateHeight = _updateTelegram->height(); - _updateTelegram->setGeometry(0, height() - updateHeight, width(), updateHeight); - scrollHeight -= updateHeight; - } - auto wasScrollHeight = _scroll->height(); - _scroll->setGeometry(0, scrollTop, width(), scrollHeight); - if (scrollHeight != wasScrollHeight) { - controller()->floatPlayerAreaUpdated().notify(true); - } - if (addToScroll) { - _scroll->scrollToY(newScrollTop); - } else { - onListScroll(); - } -} - -void DialogsWidget::updateForwardBar() { - auto selecting = App::main()->selectingPeer(); - auto oneColumnSelecting = (Adaptive::OneColumn() && selecting); - if (!oneColumnSelecting == !_forwardCancel) { - return; - } - if (oneColumnSelecting) { - _forwardCancel.create(this, st::dialogsForwardCancel); - _forwardCancel->setClickedCallback([] { Global::RefPeerChooseCancel().notify(true); }); - if (!_a_show.animating()) _forwardCancel->show(); - } else { - _forwardCancel.destroyDelayed(); - } - updateControlsGeometry(); - update(); -} - -void DialogsWidget::keyPressEvent(QKeyEvent *e) { - if (e->key() == Qt::Key_Escape) { - e->ignore(); - } else if (e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter) { - if (!_inner->choosePeer()) { - if (_inner->state() == DialogsInner::DefaultState || _inner->state() == DialogsInner::SearchedState || (_inner->state() == DialogsInner::FilteredState && _inner->hasFilteredResults())) { - _inner->selectSkip(1); - _inner->choosePeer(); - } else { - onSearchMessages(); - } - } - } else if (e->key() == Qt::Key_Down) { - _inner->setMouseSelection(false); - _inner->selectSkip(1); - } else if (e->key() == Qt::Key_Up) { - _inner->setMouseSelection(false); - _inner->selectSkip(-1); - } else if (e->key() == Qt::Key_PageDown) { - _inner->setMouseSelection(false); - _inner->selectSkipPage(_scroll->height(), 1); - } else if (e->key() == Qt::Key_PageUp) { - _inner->setMouseSelection(false); - _inner->selectSkipPage(_scroll->height(), -1); - } else { - e->ignore(); - } -} - -void DialogsWidget::paintEvent(QPaintEvent *e) { - if (App::wnd() && App::wnd()->contentOverlapped(this, e)) return; - - Painter p(this); - QRect r(e->rect()); - if (r != rect()) { - p.setClipRect(r); - } - auto progress = _a_show.current(getms(), 1.); - if (_a_show.animating()) { - auto retina = cIntRetinaFactor(); - auto fromLeft = (_showDirection == Window::SlideDirection::FromLeft); - auto coordUnder = fromLeft ? anim::interpolate(-st::slideShift, 0, progress) : anim::interpolate(0, -st::slideShift, progress); - auto coordOver = fromLeft ? anim::interpolate(0, width(), progress) : anim::interpolate(width(), 0, progress); - auto shadow = fromLeft ? (1. - progress) : progress; - if (coordOver > 0) { - p.drawPixmap(QRect(0, 0, coordOver, _cacheUnder.height() / retina), _cacheUnder, QRect(-coordUnder * retina, 0, coordOver * retina, _cacheUnder.height())); - p.setOpacity(shadow); - p.fillRect(0, 0, coordOver, _cacheUnder.height() / retina, st::slideFadeOutBg); - p.setOpacity(1); - } - p.drawPixmap(QRect(coordOver, 0, _cacheOver.width() / retina, _cacheOver.height() / retina), _cacheOver, QRect(0, 0, _cacheOver.width(), _cacheOver.height())); - p.setOpacity(shadow); - st::slideShadow.fill(p, QRect(coordOver - st::slideShadow.width(), 0, st::slideShadow.width(), _cacheOver.height() / retina)); - return; - } - auto aboveTop = 0; - if (_forwardCancel) { - p.fillRect(0, aboveTop, width(), st::dialogsForwardHeight, st::dialogsForwardBg); - p.setPen(st::dialogsForwardFg); - p.setFont(st::dialogsForwardFont); - p.drawTextLeft(st::dialogsForwardTextLeft, st::dialogsForwardTextTop, width(), lang(lng_forward_choose)); - aboveTop += st::dialogsForwardHeight; - } - auto above = QRect(0, aboveTop, width(), _scroll->y() - aboveTop); - if (above.intersects(r)) { - p.fillRect(above.intersected(r), st::dialogsBg); - } - - auto belowTop = _scroll->y() + qMin(_scroll->height(), _inner->height()); - if (!_widthAnimationCache.isNull()) { - p.drawPixmapLeft(0, _scroll->y(), width(), _widthAnimationCache); - belowTop = _scroll->y() + (_widthAnimationCache.height() / cIntRetinaFactor()); - } - - auto below = QRect(0, belowTop, width(), height() - belowTop); - if (below.intersects(r)) { - p.fillRect(below.intersected(r), st::dialogsBg); - } -} - -void DialogsWidget::destroyData() { - _inner->destroyData(); -} - -void DialogsWidget::peerBefore(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const { - return _inner->peerBefore(inPeer, inMsg, outPeer, outMsg); -} - -void DialogsWidget::peerAfter(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const { - return _inner->peerAfter(inPeer, inMsg, outPeer, outMsg); -} - -void DialogsWidget::scrollToPeer(const PeerId &peer, MsgId msgId) { - _inner->scrollToPeer(peer, msgId); -} - -void DialogsWidget::removeDialog(History *history) { - _inner->removeDialog(history); - onFilterUpdate(); -} - -Dialogs::IndexedList *DialogsWidget::contactsList() { - return _inner->contactsList(); -} - -Dialogs::IndexedList *DialogsWidget::dialogsList() { - return _inner->dialogsList(); -} - -Dialogs::IndexedList *DialogsWidget::contactsNoDialogsList() { - return _inner->contactsNoDialogsList(); -} - -bool DialogsWidget::onCancelSearch() { - bool clearing = !_filter->getLastText().isEmpty(); - if (_searchRequest) { - MTP::cancel(_searchRequest); - _searchRequest = 0; - } - if (_searchInPeer && !clearing) { - if (Adaptive::OneColumn()) { - Ui::showPeerHistory(_searchInPeer, ShowAtUnreadMsgId); - } - setSearchInPeer(nullptr); - clearing = true; - } - _inner->clearFilter(); - _filter->clear(); - _filter->updatePlaceholder(); - onFilterUpdate(); - return clearing; -} - -void DialogsWidget::onCancelSearchInPeer() { - if (_searchRequest) { - MTP::cancel(_searchRequest); - _searchRequest = 0; - } - if (_searchInPeer) { - if (Adaptive::OneColumn() && !App::main()->selectingPeer()) { - Ui::showPeerHistory(_searchInPeer, ShowAtUnreadMsgId); - } - setSearchInPeer(nullptr); - } - _inner->clearFilter(); - _filter->clear(); - _filter->updatePlaceholder(); - onFilterUpdate(); - if (!Adaptive::OneColumn() && !App::main()->selectingPeer()) { - emit cancelled(); - } -} - -void DialogsWidget::onDialogMoved(int movedFrom, int movedTo) { - int32 st = _scroll->scrollTop(); - if (st > movedTo && st < movedFrom) { - _scroll->scrollToY(st + st::dialogsRowHeight); - } -} diff --git a/Telegram/SourceFiles/dialogswidget.h b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h similarity index 60% rename from Telegram/SourceFiles/dialogswidget.h rename to Telegram/SourceFiles/dialogs/dialogs_inner_widget.h index 3fb3dc15e..fbb8685cf 100644 --- a/Telegram/SourceFiles/dialogswidget.h +++ b/Telegram/SourceFiles/dialogs/dialogs_inner_widget.h @@ -20,8 +20,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #pragma once -#include "window/section_widget.h" -#include "ui/widgets/scroll_area.h" +#include "dialogs/dialogs_widget.h" namespace Dialogs { class Row; @@ -32,28 +31,13 @@ class IndexedList; namespace Ui { class IconButton; class PopupMenu; -class DropdownMenu; -class FlatButton; class LinkButton; -class FlatInput; -class CrossButton; -template -class WidgetScaledFadeWrap; } // namespace Ui namespace Window { class Controller; } // namespace Window -enum DialogsSearchRequestType { - DialogsSearchFromStart, - DialogsSearchFromOffset, - DialogsSearchPeerFromStart, - DialogsSearchPeerFromOffset, - DialogsSearchMigratedFromStart, - DialogsSearchMigratedFromOffset, -}; - class DialogsInner : public Ui::SplittedWidget, public RPCSender, private base::Subscriber { Q_OBJECT @@ -129,7 +113,7 @@ public: ~DialogsInner(); -public slots: + public slots: void onParentGeometryChanged(); void onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars); void onPeerPhotoChanged(PeerData *peer); @@ -184,7 +168,7 @@ private: return _importantSwitchPressed || _pressed || (_hashtagPressed >= 0) || (_filteredPressed >= 0) || (_peerSearchPressed >= 0) || (_searchedPressed >= 0); } bool isSelected() const { - return _importantSwitchSelected || _selected || (_hashtagSelected >= 0) || (_filteredSelected>= 0) || (_peerSearchSelected >= 0) || (_searchedSelected >= 0); + return _importantSwitchSelected || _selected || (_hashtagSelected >= 0) || (_filteredSelected >= 0) || (_peerSearchSelected >= 0) || (_searchedSelected >= 0); } void itemRemoved(HistoryItem *item); @@ -304,174 +288,3 @@ private: }; Q_DECLARE_OPERATORS_FOR_FLAGS(DialogsInner::UpdateRowSections); - -class DialogsWidget : public Window::AbstractSectionWidget, public RPCSender { - Q_OBJECT - -public: - DialogsWidget(QWidget *parent, gsl::not_null controller); - - void updateDragInScroll(bool inScroll); - - void searchInPeer(PeerData *peer); - - void loadDialogs(); - void loadPinnedDialogs(); - void createDialog(History *history); - void dlgUpdated(Dialogs::Mode list, Dialogs::Row *row); - void dlgUpdated(PeerData *peer, MsgId msgId); - - void dialogsToUp(); - - void startWidthAnimation(); - void stopWidthAnimation(); - - bool hasTopBarShadow() const { - return true; - } - void showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms); - void showFast(); - - void destroyData(); - - void peerBefore(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const; - void peerAfter(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const; - void scrollToPeer(const PeerId &peer, MsgId msgId); - - void removeDialog(History *history); - - Dialogs::IndexedList *contactsList(); - Dialogs::IndexedList *dialogsList(); - Dialogs::IndexedList *contactsNoDialogsList(); - - void searchMessages(const QString &query, PeerData *inPeer = 0); - void onSearchMore(); - - // Float player interface. - bool wheelEventFromFloatPlayer(QEvent *e, Window::Column myColumn, Window::Column playerColumn) override; - QRect rectForFloatPlayer(Window::Column myColumn, Window::Column playerColumn) override; - - void notify_userIsContactChanged(UserData *user, bool fromThisApp); - void notify_historyMuteUpdated(History *history); - -signals: - void cancelled(); - -public slots: - void onDraggingScrollDelta(int delta); - - void onCancel(); - void onListScroll(); - void activate(); - void onFilterUpdate(bool force = false); - bool onCancelSearch(); - void onCancelSearchInPeer(); - - void onFilterCursorMoved(int from = -1, int to = -1); - void onCompleteHashtag(QString tag); - - void onDialogMoved(int movedFrom, int movedTo); - bool onSearchMessages(bool searchCache = false); - void onNeedSearchMessages(); - - void onChooseByDrag(); - -private slots: - void onDraggingScrollTimer(); - -#ifndef TDESKTOP_DISABLE_AUTOUPDATE - void onCheckUpdateStatus(); -#endif // TDESKTOP_DISABLE_AUTOUPDATE - -protected: - void dragEnterEvent(QDragEnterEvent *e) override; - void dragMoveEvent(QDragMoveEvent *e) override; - void dragLeaveEvent(QDragLeaveEvent *e) override; - void dropEvent(QDropEvent *e) override; - void resizeEvent(QResizeEvent *e) override; - void keyPressEvent(QKeyEvent *e) override; - void paintEvent(QPaintEvent *e) override; - -private: - void animationCallback(); - void dialogsReceived(const MTPmessages_Dialogs &dialogs, mtpRequestId requestId); - void pinnedDialogsReceived(const MTPmessages_PeerDialogs &dialogs, mtpRequestId requestId); - void contactsReceived(const MTPcontacts_Contacts &result); - void searchReceived(DialogsSearchRequestType type, const MTPmessages_Messages &result, mtpRequestId requestId); - void peerSearchReceived(const MTPcontacts_Found &result, mtpRequestId requestId); - - void setSearchInPeer(PeerData *peer); - void showSearchFrom(); - void showMainMenu(); - void updateLockUnlockVisibility(); - void updateJumpToDateVisibility(bool fast = false); - void updateControlsGeometry(); - void updateForwardBar(); - - void unreadCountsReceived(const QVector &dialogs); - bool dialogsFailed(const RPCError &error, mtpRequestId req); - bool contactsFailed(const RPCError &error); - bool searchFailed(DialogsSearchRequestType type, const RPCError &error, mtpRequestId req); - bool peopleFailed(const RPCError &error, mtpRequestId req); - - bool _dragInScroll = false; - bool _dragForward = false; - QTimer _chooseByDragTimer; - - bool _dialogsFull = false; - int32 _dialogsOffsetDate = 0; - MsgId _dialogsOffsetId = 0; - PeerData *_dialogsOffsetPeer = nullptr; - mtpRequestId _dialogsRequestId = 0; - mtpRequestId _pinnedDialogsRequestId = 0; - mtpRequestId _contactsRequestId = 0; - bool _pinnedDialogsReceived = false; - - object_ptr _forwardCancel = { nullptr }; - object_ptr _mainMenuToggle; - object_ptr _filter; - object_ptr> _searchFromUser; - object_ptr> _jumpToDate; - object_ptr _cancelSearch; - object_ptr _lockUnlock; - object_ptr _scroll; - QPointer _inner; - class UpdateButton; - object_ptr _updateTelegram = { nullptr }; - - Animation _a_show; - Window::SlideDirection _showDirection; - QPixmap _cacheUnder, _cacheOver; - - PeerData *_searchInPeer = nullptr; - PeerData *_searchInMigrated = nullptr; - - QTimer _searchTimer; - - QString _peerSearchQuery; - bool _peerSearchFull = false; - mtpRequestId _peerSearchRequest = 0; - - QString _searchQuery; - bool _searchFull = false; - bool _searchFullMigrated = false; - mtpRequestId _searchRequest = 0; - - using SearchCache = QMap; - SearchCache _searchCache; - - using SearchQueries = QMap; - SearchQueries _searchQueries; - - using PeerSearchCache = QMap; - PeerSearchCache _peerSearchCache; - - using PeerSearchQueries = QMap; - PeerSearchQueries _peerSearchQueries; - - QPixmap _widthAnimationCache; - - object_ptr _draggingScrollTimer = { nullptr }; - int _draggingScrollDelta = 0; - -}; diff --git a/Telegram/SourceFiles/dialogs/dialogs_widget.cpp b/Telegram/SourceFiles/dialogs/dialogs_widget.cpp new file mode 100644 index 000000000..466dc607c --- /dev/null +++ b/Telegram/SourceFiles/dialogs/dialogs_widget.cpp @@ -0,0 +1,1113 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#include "dialogs/dialogs_widget.h" + +#include "dialogs/dialogs_inner_widget.h" +#include "styles/style_dialogs.h" +#include "ui/widgets/buttons.h" +#include "lang/lang_keys.h" +#include "application.h" +#include "mainwindow.h" +#include "mainwidget.h" +#include "ui/widgets/input_fields.h" +#include "autoupdater.h" +#include "auth_session.h" +#include "messenger.h" +#include "ui/effects/widget_fade_wrap.h" +#include "window/window_controller.h" + +class DialogsWidget::UpdateButton : public Ui::RippleButton { +public: + UpdateButton(QWidget *parent); + +protected: + void paintEvent(QPaintEvent *e) override; + + void onStateChanged(State was, StateChangeSource source) override; + +private: + QString _text; + const style::FlatButton &_st; + +}; + +DialogsWidget::UpdateButton::UpdateButton(QWidget *parent) : RippleButton(parent, st::dialogsUpdateButton.ripple) +, _text(lang(lng_update_telegram).toUpper()) +, _st(st::dialogsUpdateButton) { + resize(st::dialogsWidthMin, _st.height); +} + +void DialogsWidget::UpdateButton::onStateChanged(State was, StateChangeSource source) { + RippleButton::onStateChanged(was, source); + update(); +} + +void DialogsWidget::UpdateButton::paintEvent(QPaintEvent *e) { + QPainter p(this); + + QRect r(0, height() - _st.height, width(), _st.height); + p.fillRect(r, isOver() ? _st.overBgColor : _st.bgColor); + + paintRipple(p, 0, 0, getms()); + + p.setFont(isOver() ? _st.overFont : _st.font); + p.setRenderHint(QPainter::TextAntialiasing); + p.setPen(isOver() ? _st.overColor : _st.color); + + if (width() >= st::dialogsWidthMin) { + r.setTop(_st.textTop); + p.drawText(r, _text, style::al_top); + } else { + (isOver() ? st::dialogsInstallUpdateOver : st::dialogsInstallUpdate).paintInCenter(p, r); + } +} + +DialogsWidget::DialogsWidget(QWidget *parent, gsl::not_null controller) : Window::AbstractSectionWidget(parent, controller) +, _mainMenuToggle(this, st::dialogsMenuToggle) +, _filter(this, st::dialogsFilter, langFactory(lng_dlg_filter)) +, _searchFromUser(this, object_ptr(this, st::dialogsSearchFrom)) +, _jumpToDate(this, object_ptr(this, st::dialogsCalendar)) +, _cancelSearch(this, st::dialogsCancelSearch) +, _lockUnlock(this, st::dialogsLock) +, _scroll(this, st::dialogsScroll) { + _inner = _scroll->setOwnedWidget(object_ptr(this, controller, parent)); + connect(_inner, SIGNAL(draggingScrollDelta(int)), this, SLOT(onDraggingScrollDelta(int))); + connect(_inner, SIGNAL(mustScrollTo(int,int)), _scroll, SLOT(scrollToY(int,int))); + connect(_inner, SIGNAL(dialogMoved(int,int)), this, SLOT(onDialogMoved(int,int))); + connect(_inner, SIGNAL(searchMessages()), this, SLOT(onNeedSearchMessages())); + connect(_inner, SIGNAL(searchResultChosen()), this, SLOT(onCancel())); + connect(_inner, SIGNAL(completeHashtag(QString)), this, SLOT(onCompleteHashtag(QString))); + connect(_inner, SIGNAL(refreshHashtags()), this, SLOT(onFilterCursorMoved())); + connect(_inner, SIGNAL(cancelSearchInPeer()), this, SLOT(onCancelSearchInPeer())); + connect(_scroll, SIGNAL(geometryChanged()), _inner, SLOT(onParentGeometryChanged())); + connect(_scroll, SIGNAL(scrolled()), this, SLOT(onListScroll())); + connect(_filter, SIGNAL(cancelled()), this, SLOT(onCancel())); + connect(_filter, SIGNAL(changed()), this, SLOT(onFilterUpdate())); + connect(_filter, SIGNAL(cursorPositionChanged(int,int)), this, SLOT(onFilterCursorMoved(int,int))); + +#ifndef TDESKTOP_DISABLE_AUTOUPDATE + Sandbox::connect(SIGNAL(updateLatest()), this, SLOT(onCheckUpdateStatus())); + Sandbox::connect(SIGNAL(updateFailed()), this, SLOT(onCheckUpdateStatus())); + Sandbox::connect(SIGNAL(updateReady()), this, SLOT(onCheckUpdateStatus())); + onCheckUpdateStatus(); +#endif // !TDESKTOP_DISABLE_AUTOUPDATE + + subscribe(Adaptive::Changed(), [this] { updateForwardBar(); }); + + _cancelSearch->setClickedCallback([this] { onCancelSearch(); }); + _jumpToDate->entity()->setClickedCallback([this] { if (_searchInPeer) this->controller()->showJumpToDate(_searchInPeer, QDate()); }); + _searchFromUser->entity()->setClickedCallback([this] { if (_searchInPeer->isChat() || _searchInPeer->isMegagroup()) showSearchFrom(); }); + _lockUnlock->setVisible(Global::LocalPasscode()); + subscribe(Global::RefLocalPasscodeChanged(), [this] { updateLockUnlockVisibility(); }); + _lockUnlock->setClickedCallback([this] { + _lockUnlock->setIconOverride(&st::dialogsUnlockIcon, &st::dialogsUnlockIconOver); + Messenger::Instance().setupPasscode(); + _lockUnlock->setIconOverride(nullptr); + }); + _mainMenuToggle->setClickedCallback([this] { showMainMenu(); }); + + _chooseByDragTimer.setSingleShot(true); + connect(&_chooseByDragTimer, SIGNAL(timeout()), this, SLOT(onChooseByDrag())); + + setAcceptDrops(true); + + _searchTimer.setSingleShot(true); + connect(&_searchTimer, SIGNAL(timeout()), this, SLOT(onSearchMessages())); + + _inner->setLoadMoreCallback([this] { + if (_inner->state() == DialogsInner::SearchedState || (_inner->state() == DialogsInner::FilteredState && _searchInMigrated && _searchFull && !_searchFullMigrated)) { + onSearchMore(); + } else { + loadDialogs(); + } + }); + + _filter->setFocusPolicy(Qt::StrongFocus); + _filter->customUpDown(true); + + updateJumpToDateVisibility(true); +} + +#ifndef TDESKTOP_DISABLE_AUTOUPDATE +void DialogsWidget::onCheckUpdateStatus() { + if (Sandbox::updatingState() == Application::UpdatingReady) { + if (_updateTelegram) return; + _updateTelegram.create(this); + _updateTelegram->show(); + _updateTelegram->setClickedCallback([] { + checkReadyUpdate(); + App::restart(); + }); + } else { + if (!_updateTelegram) return; + _updateTelegram.destroy(); + } + updateControlsGeometry(); +} +#endif // TDESKTOP_DISABLE_AUTOUPDATE + +void DialogsWidget::activate() { + _filter->setFocus(); + _inner->activate(); +} + +void DialogsWidget::createDialog(History *history) { + auto creating = !history->inChatList(Dialogs::Mode::All); + _inner->createDialog(history); + if (creating && history->peer->migrateFrom()) { + if (auto migrated = App::historyLoaded(history->peer->migrateFrom()->id)) { + if (migrated->inChatList(Dialogs::Mode::All)) { + removeDialog(migrated); + } + } + } +} + +void DialogsWidget::dlgUpdated(Dialogs::Mode list, Dialogs::Row *row) { + _inner->dlgUpdated(list, row); +} + +void DialogsWidget::dlgUpdated(PeerData *peer, MsgId msgId) { + _inner->dlgUpdated(peer, msgId); +} + +void DialogsWidget::dialogsToUp() { + if (_filter->getLastText().trimmed().isEmpty()) { + _scroll->scrollToY(0); + } +} + +void DialogsWidget::startWidthAnimation() { + if (!_widthAnimationCache.isNull()) { + return; + } + auto scrollGeometry = _scroll->geometry(); + auto grabGeometry = QRect(scrollGeometry.x(), scrollGeometry.y(), st::dialogsWidthMin, scrollGeometry.height()); + _scroll->setGeometry(grabGeometry); + myEnsureResized(_scroll); + auto image = QImage(grabGeometry.size() * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); + image.setDevicePixelRatio(cRetinaFactor()); + image.fill(Qt::transparent); + _scroll->render(&image, QPoint(0, 0), QRect(QPoint(0, 0), grabGeometry.size()), QWidget::DrawChildren | QWidget::IgnoreMask); + _widthAnimationCache = App::pixmapFromImageInPlace(std::move(image)); + _scroll->setGeometry(scrollGeometry); + _scroll->hide(); +} + +void DialogsWidget::stopWidthAnimation() { + _widthAnimationCache = QPixmap(); + if (!_a_show.animating()) { + _scroll->show(); + } + update(); +} + +void DialogsWidget::showFast() { + show(); + updateForwardBar(); +} + +void DialogsWidget::showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms) { + _showDirection = direction; + + _a_show.finish(); + + _cacheUnder = params.oldContentCache; + show(); + updateForwardBar(); + _cacheOver = App::main()->grabForShowAnimation(params); + + _scroll->hide(); + _mainMenuToggle->hide(); + if (_forwardCancel) _forwardCancel->hide(); + _filter->hide(); + _cancelSearch->hideFast(); + _jumpToDate->hideFast(); + _searchFromUser->hideFast(); + _lockUnlock->hide(); + + int delta = st::slideShift; + if (_showDirection == Window::SlideDirection::FromLeft) { + std::swap(_cacheUnder, _cacheOver); + } + _a_show.start([this] { animationCallback(); }, 0., 1., st::slideDuration, Window::SlideAnimation::transition()); +} + +bool DialogsWidget::wheelEventFromFloatPlayer(QEvent *e, Window::Column myColumn, Window::Column playerColumn) { + return _scroll->viewportEvent(e); +} + +QRect DialogsWidget::rectForFloatPlayer(Window::Column myColumn, Window::Column playerColumn) { + return mapToGlobal(_scroll->geometry()); +} + +void DialogsWidget::animationCallback() { + update(); + if (!_a_show.animating()) { + _cacheUnder = _cacheOver = QPixmap(); + + _scroll->show(); + _mainMenuToggle->show(); + if (_forwardCancel) _forwardCancel->show(); + _filter->show(); + updateLockUnlockVisibility(); + updateJumpToDateVisibility(true); + + onFilterUpdate(); + if (App::wnd()) App::wnd()->setInnerFocus(); + } +} + +void DialogsWidget::onCancel() { + if (!onCancelSearch() || (!_searchInPeer && !App::main()->selectingPeer())) { + emit cancelled(); + } +} + +void DialogsWidget::notify_userIsContactChanged(UserData *user, bool fromThisApp) { + if (fromThisApp) { + _filter->setText(QString()); + _filter->updatePlaceholder(); + onFilterUpdate(); + } + _inner->notify_userIsContactChanged(user, fromThisApp); +} + +void DialogsWidget::notify_historyMuteUpdated(History *history) { + _inner->notify_historyMuteUpdated(history); +} + +void DialogsWidget::unreadCountsReceived(const QVector &dialogs) { +} + +void DialogsWidget::dialogsReceived(const MTPmessages_Dialogs &dialogs, mtpRequestId requestId) { + if (_dialogsRequestId != requestId) return; + + const QVector *dialogsList = 0; + const QVector *messagesList = 0; + switch (dialogs.type()) { + case mtpc_messages_dialogs: { + auto &data = dialogs.c_messages_dialogs(); + App::feedUsers(data.vusers); + App::feedChats(data.vchats); + messagesList = &data.vmessages.v; + dialogsList = &data.vdialogs.v; + _dialogsFull = true; + } break; + case mtpc_messages_dialogsSlice: { + auto &data = dialogs.c_messages_dialogsSlice(); + App::feedUsers(data.vusers); + App::feedChats(data.vchats); + messagesList = &data.vmessages.v; + dialogsList = &data.vdialogs.v; + } break; + } + + if (!AuthSession::Current().data().contactsLoaded().value() && !_contactsRequestId) { + _contactsRequestId = MTP::send(MTPcontacts_GetContacts(MTP_string("")), rpcDone(&DialogsWidget::contactsReceived), rpcFail(&DialogsWidget::contactsFailed)); + } + + if (dialogsList) { + TimeId lastDate = 0; + PeerId lastPeer = 0; + MsgId lastMsgId = 0; + for (int i = dialogsList->size(); i > 0;) { + auto &dialog = dialogsList->at(--i); + if (dialog.type() != mtpc_dialog) { + continue; + } + + auto &dialogData = dialog.c_dialog(); + if (auto peer = peerFromMTP(dialogData.vpeer)) { + auto history = App::history(peer); + history->setPinnedDialog(dialogData.is_pinned()); + + if (!lastDate) { + if (!lastPeer) lastPeer = peer; + if (auto msgId = dialogData.vtop_message.v) { + if (!lastMsgId) lastMsgId = msgId; + for (int j = messagesList->size(); j > 0;) { + auto &message = messagesList->at(--j); + if (idFromMessage(message) == msgId && peerFromMessage(message) == peer) { + if (auto date = dateFromMessage(message)) { + lastDate = date; + } + break; + } + } + } + } + } + } + if (lastDate) { + _dialogsOffsetDate = lastDate; + _dialogsOffsetId = lastMsgId; + _dialogsOffsetPeer = App::peer(lastPeer); + } else { + _dialogsFull = true; + } + + t_assert(messagesList != nullptr); + App::feedMsgs(*messagesList, NewMessageLast); + + unreadCountsReceived(*dialogsList); + _inner->dialogsReceived(*dialogsList); + onListScroll(); + } else { + _dialogsFull = true; + } + + _dialogsRequestId = 0; + loadDialogs(); + + AuthSession::Current().data().moreChatsLoaded().notify(); + if (_dialogsFull) { + AuthSession::Current().data().allChatsLoaded().set(true); + } +} + +void DialogsWidget::pinnedDialogsReceived(const MTPmessages_PeerDialogs &dialogs, mtpRequestId requestId) { + if (_pinnedDialogsRequestId != requestId) return; + + if (dialogs.type() == mtpc_messages_peerDialogs) { + App::histories().clearPinned(); + + auto &dialogsData = dialogs.c_messages_peerDialogs(); + App::feedUsers(dialogsData.vusers); + App::feedChats(dialogsData.vchats); + auto &list = dialogsData.vdialogs.v; + for (auto i = list.size(); i > 0;) { + auto &dialog = list[--i]; + if (dialog.type() != mtpc_dialog) { + continue; + } + + auto &dialogData = dialog.c_dialog(); + if (auto peer = peerFromMTP(dialogData.vpeer)) { + auto history = App::history(peer); + history->setPinnedDialog(dialogData.is_pinned()); + } + } + App::feedMsgs(dialogsData.vmessages, NewMessageLast); + unreadCountsReceived(list); + _inner->dialogsReceived(list); + onListScroll(); + } + + _pinnedDialogsRequestId = 0; + _pinnedDialogsReceived = true; + + AuthSession::Current().data().moreChatsLoaded().notify(); +} + +bool DialogsWidget::dialogsFailed(const RPCError &error, mtpRequestId requestId) { + if (MTP::isDefaultHandledError(error)) return false; + + LOG(("RPC Error: %1 %2: %3").arg(error.code()).arg(error.type()).arg(error.description())); + if (_dialogsRequestId == requestId) { + _dialogsRequestId = 0; + } else if (_pinnedDialogsRequestId == requestId) { + _pinnedDialogsRequestId = 0; + } + return true; +} + +void DialogsWidget::onDraggingScrollDelta(int delta) { + _draggingScrollDelta = _scroll ? delta : 0; + if (_draggingScrollDelta) { + if (!_draggingScrollTimer) { + _draggingScrollTimer.create(this); + _draggingScrollTimer->setSingleShot(false); + connect(_draggingScrollTimer, SIGNAL(timeout()), this, SLOT(onDraggingScrollTimer())); + } + _draggingScrollTimer->start(15); + } else { + _draggingScrollTimer.destroy(); + } +} + +void DialogsWidget::onDraggingScrollTimer() { + auto delta = (_draggingScrollDelta > 0) ? qMin(_draggingScrollDelta * 3 / 20 + 1, int32(MaxScrollSpeed)) : qMax(_draggingScrollDelta * 3 / 20 - 1, -int32(MaxScrollSpeed)); + _scroll->scrollToY(_scroll->scrollTop() + delta); +} + +bool DialogsWidget::onSearchMessages(bool searchCache) { + QString q = _filter->getLastText().trimmed(); + if (q.isEmpty()) { + MTP::cancel(base::take(_searchRequest)); + MTP::cancel(base::take(_peerSearchRequest)); + return true; + } + if (searchCache) { + SearchCache::const_iterator i = _searchCache.constFind(q); + if (i != _searchCache.cend()) { + _searchQuery = q; + _searchFull = _searchFullMigrated = false; + MTP::cancel(base::take(_searchRequest)); + searchReceived(_searchInPeer ? DialogsSearchPeerFromStart : DialogsSearchFromStart, i.value(), 0); + return true; + } + } else if (_searchQuery != q) { + _searchQuery = q; + _searchFull = _searchFullMigrated = false; + MTP::cancel(base::take(_searchRequest)); + if (_searchInPeer) { + _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInPeer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, DialogsSearchPeerFromStart), rpcFail(&DialogsWidget::searchFailed, DialogsSearchPeerFromStart)); + } else { + _searchRequest = MTP::send(MTPmessages_SearchGlobal(MTP_string(_searchQuery), MTP_int(0), MTP_inputPeerEmpty(), MTP_int(0), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, DialogsSearchFromStart), rpcFail(&DialogsWidget::searchFailed, DialogsSearchFromStart)); + } + _searchQueries.insert(_searchRequest, _searchQuery); + } + if (!_searchInPeer && q.size() >= MinUsernameLength) { + if (searchCache) { + auto i = _peerSearchCache.constFind(q); + if (i != _peerSearchCache.cend()) { + _peerSearchQuery = q; + _peerSearchRequest = 0; + peerSearchReceived(i.value(), 0); + return true; + } + } else if (_peerSearchQuery != q) { + _peerSearchQuery = q; + _peerSearchFull = false; + _peerSearchRequest = MTP::send(MTPcontacts_Search(MTP_string(_peerSearchQuery), MTP_int(SearchPeopleLimit)), rpcDone(&DialogsWidget::peerSearchReceived), rpcFail(&DialogsWidget::peopleFailed)); + _peerSearchQueries.insert(_peerSearchRequest, _peerSearchQuery); + } + } + return false; +} + +void DialogsWidget::onNeedSearchMessages() { + if (!onSearchMessages(true)) { + _searchTimer.start(AutoSearchTimeout); + } +} + +void DialogsWidget::onChooseByDrag() { + _inner->choosePeer(); +} + +void DialogsWidget::showMainMenu() { + App::wnd()->showMainMenu(); +} + +void DialogsWidget::searchMessages(const QString &query, PeerData *inPeer) { + if ((_filter->getLastText() != query) || (inPeer && inPeer != _searchInPeer && inPeer->migrateTo() != _searchInPeer)) { + if (inPeer) { + onCancelSearch(); + setSearchInPeer(inPeer); + } + _filter->setText(query); + _filter->updatePlaceholder(); + onFilterUpdate(true); + _searchTimer.stop(); + onSearchMessages(); + + _inner->saveRecentHashtags(query); + } +} + +void DialogsWidget::onSearchMore() { + if (!_searchRequest) { + if (!_searchFull) { + auto offsetDate = _inner->lastSearchDate(); + auto offsetPeer = _inner->lastSearchPeer(); + auto offsetId = _inner->lastSearchId(); + if (_searchInPeer) { + _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInPeer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(offsetId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetId ? DialogsSearchPeerFromOffset : DialogsSearchPeerFromStart), rpcFail(&DialogsWidget::searchFailed, offsetId ? DialogsSearchPeerFromOffset : DialogsSearchPeerFromStart)); + } else { + _searchRequest = MTP::send(MTPmessages_SearchGlobal(MTP_string(_searchQuery), MTP_int(offsetDate), offsetPeer ? offsetPeer->input : MTP_inputPeerEmpty(), MTP_int(offsetId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetId ? DialogsSearchFromOffset : DialogsSearchFromStart), rpcFail(&DialogsWidget::searchFailed, offsetId ? DialogsSearchFromOffset : DialogsSearchFromStart)); + } + if (!offsetId) { + _searchQueries.insert(_searchRequest, _searchQuery); + } + } else if (_searchInMigrated && !_searchFullMigrated) { + auto offsetMigratedId = _inner->lastSearchMigratedId(); + _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _searchInMigrated->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), MTP_inputMessagesFilterEmpty(), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(offsetMigratedId), MTP_int(SearchPerPage)), rpcDone(&DialogsWidget::searchReceived, offsetMigratedId ? DialogsSearchMigratedFromOffset : DialogsSearchMigratedFromStart), rpcFail(&DialogsWidget::searchFailed, offsetMigratedId ? DialogsSearchMigratedFromOffset : DialogsSearchMigratedFromStart)); + } + } +} + +void DialogsWidget::loadDialogs() { + if (_dialogsRequestId) return; + if (_dialogsFull) { + _inner->addAllSavedPeers(); + return; + } + + auto firstLoad = !_dialogsOffsetDate; + auto loadCount = firstLoad ? DialogsFirstLoad : DialogsPerPage; + auto flags = MTPmessages_GetDialogs::Flag::f_exclude_pinned; + _dialogsRequestId = MTP::send(MTPmessages_GetDialogs(MTP_flags(flags), MTP_int(_dialogsOffsetDate), MTP_int(_dialogsOffsetId), _dialogsOffsetPeer ? _dialogsOffsetPeer->input : MTP_inputPeerEmpty(), MTP_int(loadCount)), rpcDone(&DialogsWidget::dialogsReceived), rpcFail(&DialogsWidget::dialogsFailed)); + if (!_pinnedDialogsReceived) { + loadPinnedDialogs(); + } +} + +void DialogsWidget::loadPinnedDialogs() { + if (_pinnedDialogsRequestId) return; + + _pinnedDialogsReceived = false; + _pinnedDialogsRequestId = MTP::send(MTPmessages_GetPinnedDialogs(), rpcDone(&DialogsWidget::pinnedDialogsReceived), rpcFail(&DialogsWidget::dialogsFailed)); +} + +void DialogsWidget::contactsReceived(const MTPcontacts_Contacts &result) { + _contactsRequestId = 0; + if (result.type() == mtpc_contacts_contacts) { + auto &d = result.c_contacts_contacts(); + App::feedUsers(d.vusers); + _inner->contactsReceived(d.vcontacts.v); + } + AuthSession::Current().data().contactsLoaded().set(true); +} + +bool DialogsWidget::contactsFailed(const RPCError &error) { + if (MTP::isDefaultHandledError(error)) return false; + _contactsRequestId = 0; + return true; +} + +void DialogsWidget::searchReceived(DialogsSearchRequestType type, const MTPmessages_Messages &result, mtpRequestId req) { + if (_inner->state() == DialogsInner::FilteredState || _inner->state() == DialogsInner::SearchedState) { + if (type == DialogsSearchFromStart || type == DialogsSearchPeerFromStart) { + auto i = _searchQueries.find(req); + if (i != _searchQueries.cend()) { + _searchCache[i.value()] = result; + _searchQueries.erase(i); + } + } + } + + if (_searchRequest == req) { + switch (result.type()) { + case mtpc_messages_messages: { + auto &d = result.c_messages_messages(); + App::feedUsers(d.vusers); + App::feedChats(d.vchats); + auto &msgs = d.vmessages.v; + if (!_inner->searchReceived(msgs, type, msgs.size())) { + if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { + _searchFullMigrated = true; + } else { + _searchFull = true; + } + } + } break; + + case mtpc_messages_messagesSlice: { + auto &d = result.c_messages_messagesSlice(); + App::feedUsers(d.vusers); + App::feedChats(d.vchats); + auto &msgs = d.vmessages.v; + if (!_inner->searchReceived(msgs, type, d.vcount.v)) { + if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { + _searchFullMigrated = true; + } else { + _searchFull = true; + } + } + } break; + + case mtpc_messages_channelMessages: { + auto &d = result.c_messages_channelMessages(); + if (_searchInPeer && _searchInPeer->isChannel()) { + _searchInPeer->asChannel()->ptsReceived(d.vpts.v); + } else { + LOG(("API Error: received messages.channelMessages when no channel was passed! (DialogsWidget::searchReceived)")); + } + App::feedUsers(d.vusers); + App::feedChats(d.vchats); + auto &msgs = d.vmessages.v; + if (!_inner->searchReceived(msgs, type, d.vcount.v)) { + if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { + _searchFullMigrated = true; + } else { + _searchFull = true; + } + } + } break; + } + + _searchRequest = 0; + onListScroll(); + update(); + } +} + +void DialogsWidget::peerSearchReceived(const MTPcontacts_Found &result, mtpRequestId req) { + auto q = _peerSearchQuery; + if (_inner->state() == DialogsInner::FilteredState || _inner->state() == DialogsInner::SearchedState) { + auto i = _peerSearchQueries.find(req); + if (i != _peerSearchQueries.cend()) { + q = i.value(); + _peerSearchCache[q] = result; + _peerSearchQueries.erase(i); + } + } + if (_peerSearchRequest == req) { + switch (result.type()) { + case mtpc_contacts_found: { + auto &d = result.c_contacts_found(); + App::feedUsers(d.vusers); + App::feedChats(d.vchats); + _inner->peerSearchReceived(q, d.vresults.v); + } break; + } + + _peerSearchRequest = 0; + onListScroll(); + } +} + +bool DialogsWidget::searchFailed(DialogsSearchRequestType type, const RPCError &error, mtpRequestId req) { + if (MTP::isDefaultHandledError(error)) return false; + + if (_searchRequest == req) { + _searchRequest = 0; + if (type == DialogsSearchMigratedFromStart || type == DialogsSearchMigratedFromOffset) { + _searchFullMigrated = true; + } else { + _searchFull = true; + } + } + return true; +} + +bool DialogsWidget::peopleFailed(const RPCError &error, mtpRequestId req) { + if (MTP::isDefaultHandledError(error)) return false; + + if (_peerSearchRequest == req) { + _peerSearchRequest = 0; + _peerSearchFull = true; + } + return true; +} + +void DialogsWidget::dragEnterEvent(QDragEnterEvent *e) { + if (App::main()->selectingPeer()) return; + + _dragInScroll = false; + _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-selected")); + if (!_dragForward) _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-pressed-link")); + if (!_dragForward) _dragForward = e->mimeData()->hasFormat(qsl("application/x-td-forward-pressed")); + if (_dragForward && Adaptive::OneColumn()) _dragForward = false; + if (_dragForward) { + e->setDropAction(Qt::CopyAction); + e->accept(); + updateDragInScroll(_scroll->geometry().contains(e->pos())); + } else if (App::main() && App::main()->getDragState(e->mimeData()) != DragStateNone) { + e->setDropAction(Qt::CopyAction); + e->accept(); + } + _chooseByDragTimer.stop(); +} + +void DialogsWidget::dragMoveEvent(QDragMoveEvent *e) { + if (_scroll->geometry().contains(e->pos())) { + if (_dragForward) { + updateDragInScroll(true); + } else { + _chooseByDragTimer.start(ChoosePeerByDragTimeout); + } + PeerData *p = _inner->updateFromParentDrag(mapToGlobal(e->pos())); + if (p) { + e->setDropAction(Qt::CopyAction); + } else { + e->setDropAction(Qt::IgnoreAction); + } + } else { + if (_dragForward) updateDragInScroll(false); + _inner->dragLeft(); + e->setDropAction(Qt::IgnoreAction); + } + e->accept(); +} + +void DialogsWidget::dragLeaveEvent(QDragLeaveEvent *e) { + if (_dragForward) { + updateDragInScroll(false); + } else { + _chooseByDragTimer.stop(); + } + _inner->dragLeft(); + e->accept(); +} + +void DialogsWidget::updateDragInScroll(bool inScroll) { + if (_dragInScroll != inScroll) { + _dragInScroll = inScroll; + if (_dragInScroll) { + App::main()->showForwardLayer(SelectedItemSet()); + } else { + App::main()->dialogsCancelled(); + } + } +} + +void DialogsWidget::dropEvent(QDropEvent *e) { + _chooseByDragTimer.stop(); + if (_scroll->geometry().contains(e->pos())) { + if (auto peer = _inner->updateFromParentDrag(mapToGlobal(e->pos()))) { + e->acceptProposedAction(); + App::main()->onFilesOrForwardDrop(peer->id, e->mimeData()); + } + } +} + +void DialogsWidget::onListScroll() { + auto scrollTop = _scroll->scrollTop(); + _inner->setVisibleTopBottom(scrollTop, scrollTop + _scroll->height()); +} + +void DialogsWidget::onFilterUpdate(bool force) { + if (_a_show.animating() && !force) return; + + auto filterText = _filter->getLastText(); + _inner->onFilterUpdate(filterText, force); + if (filterText.isEmpty()) { + _searchCache.clear(); + _searchQueries.clear(); + _searchQuery = QString(); + _cancelSearch->hideAnimated(); + } else { + _cancelSearch->showAnimated(); + } + updateJumpToDateVisibility(); + + if (filterText.size() < MinUsernameLength) { + _peerSearchCache.clear(); + _peerSearchQueries.clear(); + _peerSearchQuery = QString(); + } +} + +void DialogsWidget::searchInPeer(PeerData *peer) { + onCancelSearch(); + setSearchInPeer(peer); + onFilterUpdate(true); +} + +void DialogsWidget::setSearchInPeer(PeerData *peer) { + auto newSearchInPeer = peer ? (peer->migrateTo() ? peer->migrateTo() : peer) : nullptr; + _searchInMigrated = newSearchInPeer ? newSearchInPeer->migrateFrom() : nullptr; + if (newSearchInPeer != _searchInPeer) { + _searchInPeer = newSearchInPeer; + controller()->searchInPeerChanged().notify(_searchInPeer, true); + updateJumpToDateVisibility(); + } + _inner->searchInPeer(_searchInPeer); +} + +void DialogsWidget::showSearchFrom() { + +} + +void DialogsWidget::onFilterCursorMoved(int from, int to) { + if (to < 0) to = _filter->cursorPosition(); + QString t = _filter->getLastText(); + QStringRef r; + for (int start = to; start > 0;) { + --start; + if (t.size() <= start) break; + if (t.at(start) == '#') { + r = t.midRef(start, to - start); + break; + } + if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break; + } + _inner->onHashtagFilterUpdate(r); +} + +void DialogsWidget::onCompleteHashtag(QString tag) { + QString t = _filter->getLastText(), r; + int cur = _filter->cursorPosition(); + for (int start = cur; start > 0;) { + --start; + if (t.size() <= start) break; + if (t.at(start) == '#') { + if (cur == start + 1 || t.midRef(start + 1, cur - start - 1) == tag.midRef(0, cur - start - 1)) { + for (; cur < t.size() && cur - start - 1 < tag.size(); ++cur) { + if (t.at(cur) != tag.at(cur - start - 1)) break; + } + if (cur - start - 1 == tag.size() && cur < t.size() && t.at(cur) == ' ') ++cur; + r = t.mid(0, start + 1) + tag + ' ' + t.mid(cur); + _filter->setText(r); + _filter->setCursorPosition(start + 1 + tag.size() + 1); + onFilterUpdate(true); + return; + } + break; + } + if (!t.at(start).isLetterOrNumber() && t.at(start) != '_') break; + } + _filter->setText(t.mid(0, cur) + '#' + tag + ' ' + t.mid(cur)); + _filter->setCursorPosition(cur + 1 + tag.size() + 1); + onFilterUpdate(true); +} + +void DialogsWidget::resizeEvent(QResizeEvent *e) { + updateControlsGeometry(); +} + +void DialogsWidget::updateLockUnlockVisibility() { + if (!_a_show.animating()) { + _lockUnlock->setVisible(Global::LocalPasscode()); + } + updateControlsGeometry(); +} + +void DialogsWidget::updateJumpToDateVisibility(bool fast) { + if (_a_show.animating()) return; + + auto jumpToDateVisible = (_searchInPeer && _filter->getLastText().isEmpty()); + if (fast) { + _jumpToDate->toggleFast(jumpToDateVisible); + } else { + _jumpToDate->toggleAnimated(jumpToDateVisible); + } + + auto searchFromUserVisible = _searchInPeer && (_searchInPeer->isChat() || _searchInPeer->isMegagroup()); + if (fast) { + _searchFromUser->toggleFast(searchFromUserVisible); + } else { + _searchFromUser->toggleAnimated(searchFromUserVisible); + } +} + +void DialogsWidget::updateControlsGeometry() { + auto filterAreaTop = 0; + if (_forwardCancel) { + _forwardCancel->moveToLeft(0, filterAreaTop); + filterAreaTop += st::dialogsForwardHeight; + } + auto smallLayoutWidth = (st::dialogsPadding.x() + st::dialogsPhotoSize + st::dialogsPadding.x()); + auto smallLayoutRatio = (width() < st::dialogsWidthMin) ? (st::dialogsWidthMin - width()) / float64(st::dialogsWidthMin - smallLayoutWidth) : 0.; + auto filterLeft = st::dialogsFilterPadding.x() + _mainMenuToggle->width() + st::dialogsFilterPadding.x(); + auto filterRight = (Global::LocalPasscode() ? (st::dialogsFilterPadding.x() + _lockUnlock->width()) : st::dialogsFilterSkip) + st::dialogsFilterPadding.x(); + auto filterWidth = qMax(width(), st::dialogsWidthMin) - filterLeft - filterRight; + auto filterAreaHeight = st::dialogsFilterPadding.y() + _mainMenuToggle->height() + st::dialogsFilterPadding.y(); + auto filterTop = filterAreaTop + (filterAreaHeight - _filter->height()) / 2; + filterLeft = anim::interpolate(filterLeft, smallLayoutWidth, smallLayoutRatio); + _filter->setGeometryToLeft(filterLeft, filterTop, filterWidth, _filter->height()); + auto mainMenuLeft = anim::interpolate(st::dialogsFilterPadding.x(), (smallLayoutWidth - _mainMenuToggle->width()) / 2, smallLayoutRatio); + _mainMenuToggle->moveToLeft(mainMenuLeft, filterAreaTop + st::dialogsFilterPadding.y()); + _lockUnlock->moveToLeft(filterLeft + filterWidth + st::dialogsFilterPadding.x(), filterAreaTop + st::dialogsFilterPadding.y()); + _cancelSearch->moveToLeft(filterLeft + filterWidth - _cancelSearch->width(), _filter->y()); + _jumpToDate->moveToLeft(filterLeft + filterWidth - _jumpToDate->width(), _filter->y()); + _searchFromUser->moveToLeft(filterLeft + filterWidth - _jumpToDate->width() - _searchFromUser->width(), _filter->y()); + + auto scrollTop = filterAreaTop + filterAreaHeight; + auto addToScroll = App::main() ? App::main()->contentScrollAddToY() : 0; + auto newScrollTop = _scroll->scrollTop() + addToScroll; + auto scrollHeight = height() - scrollTop; + if (_updateTelegram) { + auto updateHeight = _updateTelegram->height(); + _updateTelegram->setGeometry(0, height() - updateHeight, width(), updateHeight); + scrollHeight -= updateHeight; + } + auto wasScrollHeight = _scroll->height(); + _scroll->setGeometry(0, scrollTop, width(), scrollHeight); + if (scrollHeight != wasScrollHeight) { + controller()->floatPlayerAreaUpdated().notify(true); + } + if (addToScroll) { + _scroll->scrollToY(newScrollTop); + } else { + onListScroll(); + } +} + +void DialogsWidget::updateForwardBar() { + auto selecting = App::main()->selectingPeer(); + auto oneColumnSelecting = (Adaptive::OneColumn() && selecting); + if (!oneColumnSelecting == !_forwardCancel) { + return; + } + if (oneColumnSelecting) { + _forwardCancel.create(this, st::dialogsForwardCancel); + _forwardCancel->setClickedCallback([] { Global::RefPeerChooseCancel().notify(true); }); + if (!_a_show.animating()) _forwardCancel->show(); + } else { + _forwardCancel.destroyDelayed(); + } + updateControlsGeometry(); + update(); +} + +void DialogsWidget::keyPressEvent(QKeyEvent *e) { + if (e->key() == Qt::Key_Escape) { + e->ignore(); + } else if (e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter) { + if (!_inner->choosePeer()) { + if (_inner->state() == DialogsInner::DefaultState || _inner->state() == DialogsInner::SearchedState || (_inner->state() == DialogsInner::FilteredState && _inner->hasFilteredResults())) { + _inner->selectSkip(1); + _inner->choosePeer(); + } else { + onSearchMessages(); + } + } + } else if (e->key() == Qt::Key_Down) { + _inner->setMouseSelection(false); + _inner->selectSkip(1); + } else if (e->key() == Qt::Key_Up) { + _inner->setMouseSelection(false); + _inner->selectSkip(-1); + } else if (e->key() == Qt::Key_PageDown) { + _inner->setMouseSelection(false); + _inner->selectSkipPage(_scroll->height(), 1); + } else if (e->key() == Qt::Key_PageUp) { + _inner->setMouseSelection(false); + _inner->selectSkipPage(_scroll->height(), -1); + } else { + e->ignore(); + } +} + +void DialogsWidget::paintEvent(QPaintEvent *e) { + if (App::wnd() && App::wnd()->contentOverlapped(this, e)) return; + + Painter p(this); + QRect r(e->rect()); + if (r != rect()) { + p.setClipRect(r); + } + auto progress = _a_show.current(getms(), 1.); + if (_a_show.animating()) { + auto retina = cIntRetinaFactor(); + auto fromLeft = (_showDirection == Window::SlideDirection::FromLeft); + auto coordUnder = fromLeft ? anim::interpolate(-st::slideShift, 0, progress) : anim::interpolate(0, -st::slideShift, progress); + auto coordOver = fromLeft ? anim::interpolate(0, width(), progress) : anim::interpolate(width(), 0, progress); + auto shadow = fromLeft ? (1. - progress) : progress; + if (coordOver > 0) { + p.drawPixmap(QRect(0, 0, coordOver, _cacheUnder.height() / retina), _cacheUnder, QRect(-coordUnder * retina, 0, coordOver * retina, _cacheUnder.height())); + p.setOpacity(shadow); + p.fillRect(0, 0, coordOver, _cacheUnder.height() / retina, st::slideFadeOutBg); + p.setOpacity(1); + } + p.drawPixmap(QRect(coordOver, 0, _cacheOver.width() / retina, _cacheOver.height() / retina), _cacheOver, QRect(0, 0, _cacheOver.width(), _cacheOver.height())); + p.setOpacity(shadow); + st::slideShadow.fill(p, QRect(coordOver - st::slideShadow.width(), 0, st::slideShadow.width(), _cacheOver.height() / retina)); + return; + } + auto aboveTop = 0; + if (_forwardCancel) { + p.fillRect(0, aboveTop, width(), st::dialogsForwardHeight, st::dialogsForwardBg); + p.setPen(st::dialogsForwardFg); + p.setFont(st::dialogsForwardFont); + p.drawTextLeft(st::dialogsForwardTextLeft, st::dialogsForwardTextTop, width(), lang(lng_forward_choose)); + aboveTop += st::dialogsForwardHeight; + } + auto above = QRect(0, aboveTop, width(), _scroll->y() - aboveTop); + if (above.intersects(r)) { + p.fillRect(above.intersected(r), st::dialogsBg); + } + + auto belowTop = _scroll->y() + qMin(_scroll->height(), _inner->height()); + if (!_widthAnimationCache.isNull()) { + p.drawPixmapLeft(0, _scroll->y(), width(), _widthAnimationCache); + belowTop = _scroll->y() + (_widthAnimationCache.height() / cIntRetinaFactor()); + } + + auto below = QRect(0, belowTop, width(), height() - belowTop); + if (below.intersects(r)) { + p.fillRect(below.intersected(r), st::dialogsBg); + } +} + +void DialogsWidget::destroyData() { + _inner->destroyData(); +} + +void DialogsWidget::peerBefore(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const { + return _inner->peerBefore(inPeer, inMsg, outPeer, outMsg); +} + +void DialogsWidget::peerAfter(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const { + return _inner->peerAfter(inPeer, inMsg, outPeer, outMsg); +} + +void DialogsWidget::scrollToPeer(const PeerId &peer, MsgId msgId) { + _inner->scrollToPeer(peer, msgId); +} + +void DialogsWidget::removeDialog(History *history) { + _inner->removeDialog(history); + onFilterUpdate(); +} + +Dialogs::IndexedList *DialogsWidget::contactsList() { + return _inner->contactsList(); +} + +Dialogs::IndexedList *DialogsWidget::dialogsList() { + return _inner->dialogsList(); +} + +Dialogs::IndexedList *DialogsWidget::contactsNoDialogsList() { + return _inner->contactsNoDialogsList(); +} + +bool DialogsWidget::onCancelSearch() { + bool clearing = !_filter->getLastText().isEmpty(); + if (_searchRequest) { + MTP::cancel(_searchRequest); + _searchRequest = 0; + } + if (_searchInPeer && !clearing) { + if (Adaptive::OneColumn()) { + Ui::showPeerHistory(_searchInPeer, ShowAtUnreadMsgId); + } + setSearchInPeer(nullptr); + clearing = true; + } + _inner->clearFilter(); + _filter->clear(); + _filter->updatePlaceholder(); + onFilterUpdate(); + return clearing; +} + +void DialogsWidget::onCancelSearchInPeer() { + if (_searchRequest) { + MTP::cancel(_searchRequest); + _searchRequest = 0; + } + if (_searchInPeer) { + if (Adaptive::OneColumn() && !App::main()->selectingPeer()) { + Ui::showPeerHistory(_searchInPeer, ShowAtUnreadMsgId); + } + setSearchInPeer(nullptr); + } + _inner->clearFilter(); + _filter->clear(); + _filter->updatePlaceholder(); + onFilterUpdate(); + if (!Adaptive::OneColumn() && !App::main()->selectingPeer()) { + emit cancelled(); + } +} + +void DialogsWidget::onDialogMoved(int movedFrom, int movedTo) { + int32 st = _scroll->scrollTop(); + if (st > movedTo && st < movedFrom) { + _scroll->scrollToY(st + st::dialogsRowHeight); + } +} diff --git a/Telegram/SourceFiles/dialogs/dialogs_widget.h b/Telegram/SourceFiles/dialogs/dialogs_widget.h new file mode 100644 index 000000000..90491918c --- /dev/null +++ b/Telegram/SourceFiles/dialogs/dialogs_widget.h @@ -0,0 +1,227 @@ +/* +This file is part of Telegram Desktop, +the official desktop version of Telegram messaging app, see https://telegram.org + +Telegram Desktop is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 3 of the License, or +(at your option) any later version. + +It is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +In addition, as a special exception, the copyright holders give permission +to link the code of portions of this program with the OpenSSL library. + +Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE +Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org +*/ +#pragma once + +#include "window/section_widget.h" +#include "ui/widgets/scroll_area.h" + +class DialogsInner; + +namespace Dialogs { +class Row; +class FakeRow; +class IndexedList; +} // namespace Dialogs + +namespace Ui { +class IconButton; +class PopupMenu; +class DropdownMenu; +class FlatButton; +class FlatInput; +class CrossButton; +template +class WidgetScaledFadeWrap; +} // namespace Ui + +namespace Window { +class Controller; +} // namespace Window + +enum DialogsSearchRequestType { + DialogsSearchFromStart, + DialogsSearchFromOffset, + DialogsSearchPeerFromStart, + DialogsSearchPeerFromOffset, + DialogsSearchMigratedFromStart, + DialogsSearchMigratedFromOffset, +}; + +class DialogsWidget : public Window::AbstractSectionWidget, public RPCSender { + Q_OBJECT + +public: + DialogsWidget(QWidget *parent, gsl::not_null controller); + + void updateDragInScroll(bool inScroll); + + void searchInPeer(PeerData *peer); + + void loadDialogs(); + void loadPinnedDialogs(); + void createDialog(History *history); + void dlgUpdated(Dialogs::Mode list, Dialogs::Row *row); + void dlgUpdated(PeerData *peer, MsgId msgId); + + void dialogsToUp(); + + void startWidthAnimation(); + void stopWidthAnimation(); + + bool hasTopBarShadow() const { + return true; + } + void showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms); + void showFast(); + + void destroyData(); + + void peerBefore(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const; + void peerAfter(const PeerData *inPeer, MsgId inMsg, PeerData *&outPeer, MsgId &outMsg) const; + void scrollToPeer(const PeerId &peer, MsgId msgId); + + void removeDialog(History *history); + + Dialogs::IndexedList *contactsList(); + Dialogs::IndexedList *dialogsList(); + Dialogs::IndexedList *contactsNoDialogsList(); + + void searchMessages(const QString &query, PeerData *inPeer = 0); + void onSearchMore(); + + // Float player interface. + bool wheelEventFromFloatPlayer(QEvent *e, Window::Column myColumn, Window::Column playerColumn) override; + QRect rectForFloatPlayer(Window::Column myColumn, Window::Column playerColumn) override; + + void notify_userIsContactChanged(UserData *user, bool fromThisApp); + void notify_historyMuteUpdated(History *history); + +signals: + void cancelled(); + +public slots: + void onDraggingScrollDelta(int delta); + + void onCancel(); + void onListScroll(); + void activate(); + void onFilterUpdate(bool force = false); + bool onCancelSearch(); + void onCancelSearchInPeer(); + + void onFilterCursorMoved(int from = -1, int to = -1); + void onCompleteHashtag(QString tag); + + void onDialogMoved(int movedFrom, int movedTo); + bool onSearchMessages(bool searchCache = false); + void onNeedSearchMessages(); + + void onChooseByDrag(); + +private slots: + void onDraggingScrollTimer(); + +#ifndef TDESKTOP_DISABLE_AUTOUPDATE + void onCheckUpdateStatus(); +#endif // TDESKTOP_DISABLE_AUTOUPDATE + +protected: + void dragEnterEvent(QDragEnterEvent *e) override; + void dragMoveEvent(QDragMoveEvent *e) override; + void dragLeaveEvent(QDragLeaveEvent *e) override; + void dropEvent(QDropEvent *e) override; + void resizeEvent(QResizeEvent *e) override; + void keyPressEvent(QKeyEvent *e) override; + void paintEvent(QPaintEvent *e) override; + +private: + void animationCallback(); + void dialogsReceived(const MTPmessages_Dialogs &dialogs, mtpRequestId requestId); + void pinnedDialogsReceived(const MTPmessages_PeerDialogs &dialogs, mtpRequestId requestId); + void contactsReceived(const MTPcontacts_Contacts &result); + void searchReceived(DialogsSearchRequestType type, const MTPmessages_Messages &result, mtpRequestId requestId); + void peerSearchReceived(const MTPcontacts_Found &result, mtpRequestId requestId); + + void setSearchInPeer(PeerData *peer); + void showSearchFrom(); + void showMainMenu(); + void updateLockUnlockVisibility(); + void updateJumpToDateVisibility(bool fast = false); + void updateControlsGeometry(); + void updateForwardBar(); + + void unreadCountsReceived(const QVector &dialogs); + bool dialogsFailed(const RPCError &error, mtpRequestId req); + bool contactsFailed(const RPCError &error); + bool searchFailed(DialogsSearchRequestType type, const RPCError &error, mtpRequestId req); + bool peopleFailed(const RPCError &error, mtpRequestId req); + + bool _dragInScroll = false; + bool _dragForward = false; + QTimer _chooseByDragTimer; + + bool _dialogsFull = false; + int32 _dialogsOffsetDate = 0; + MsgId _dialogsOffsetId = 0; + PeerData *_dialogsOffsetPeer = nullptr; + mtpRequestId _dialogsRequestId = 0; + mtpRequestId _pinnedDialogsRequestId = 0; + mtpRequestId _contactsRequestId = 0; + bool _pinnedDialogsReceived = false; + + object_ptr _forwardCancel = { nullptr }; + object_ptr _mainMenuToggle; + object_ptr _filter; + object_ptr> _searchFromUser; + object_ptr> _jumpToDate; + object_ptr _cancelSearch; + object_ptr _lockUnlock; + object_ptr _scroll; + QPointer _inner; + class UpdateButton; + object_ptr _updateTelegram = { nullptr }; + + Animation _a_show; + Window::SlideDirection _showDirection; + QPixmap _cacheUnder, _cacheOver; + + PeerData *_searchInPeer = nullptr; + PeerData *_searchInMigrated = nullptr; + + QTimer _searchTimer; + + QString _peerSearchQuery; + bool _peerSearchFull = false; + mtpRequestId _peerSearchRequest = 0; + + QString _searchQuery; + bool _searchFull = false; + bool _searchFullMigrated = false; + mtpRequestId _searchRequest = 0; + + using SearchCache = QMap; + SearchCache _searchCache; + + using SearchQueries = QMap; + SearchQueries _searchQueries; + + using PeerSearchCache = QMap; + PeerSearchCache _peerSearchCache; + + using PeerSearchQueries = QMap; + PeerSearchQueries _peerSearchQueries; + + QPixmap _widthAnimationCache; + + object_ptr _draggingScrollTimer = { nullptr }; + int _draggingScrollDelta = 0; + +}; diff --git a/Telegram/SourceFiles/history/history.style b/Telegram/SourceFiles/history/history.style index 026484ab9..d4bb49d9f 100644 --- a/Telegram/SourceFiles/history/history.style +++ b/Telegram/SourceFiles/history/history.style @@ -353,24 +353,6 @@ botKbTinyButton: BotKeyboardButton { } botKbScroll: defaultSolidScroll; -mentionHeight: 40px; -mentionScroll: ScrollArea(defaultScrollArea) { - topsh: 0px; - bottomsh: 0px; -} -mentionPadding: margins(8px, 5px, 8px, 5px); -mentionTop: 11px; -mentionFont: linkFont; -mentionNameFg: windowFg; -mentionNameFgOver: windowFgOver; -mentionPhotoSize: msgPhotoSize; -mentionBg: windowBg; -mentionBgOver: windowBgOver; -mentionFg: windowSubTextFg; -mentionFgOver: windowSubTextFgOver; -mentionFgActive: windowActiveTextFg; -mentionFgOverActive: windowActiveTextFg; - historyDateFadeDuration: 200; historyPhotoLeft: 14px; diff --git a/Telegram/SourceFiles/mainwidget.cpp b/Telegram/SourceFiles/mainwidget.cpp index 36da10026..9b7e4bb20 100644 --- a/Telegram/SourceFiles/mainwidget.cpp +++ b/Telegram/SourceFiles/mainwidget.cpp @@ -33,7 +33,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "chat_helpers/message_field.h" #include "observer_peer.h" #include "apiwrap.h" -#include "dialogswidget.h" +#include "dialogs/dialogs_widget.h" #include "history/history_widget.h" #include "history/history_message.h" #include "history/history_media.h" diff --git a/Telegram/gyp/telegram_sources.txt b/Telegram/gyp/telegram_sources.txt index d528bcc27..4a1b5ad94 100644 --- a/Telegram/gyp/telegram_sources.txt +++ b/Telegram/gyp/telegram_sources.txt @@ -136,12 +136,16 @@ <(src_loc)/dialogs/dialogs_common.h <(src_loc)/dialogs/dialogs_indexed_list.cpp <(src_loc)/dialogs/dialogs_indexed_list.h +<(src_loc)/dialogs/dialogs_inner_widget.cpp +<(src_loc)/dialogs/dialogs_inner_widget.h <(src_loc)/dialogs/dialogs_layout.cpp <(src_loc)/dialogs/dialogs_layout.h <(src_loc)/dialogs/dialogs_list.cpp <(src_loc)/dialogs/dialogs_list.h <(src_loc)/dialogs/dialogs_row.cpp <(src_loc)/dialogs/dialogs_row.h +<(src_loc)/dialogs/dialogs_widget.cpp +<(src_loc)/dialogs/dialogs_widget.h <(src_loc)/history/history.cpp <(src_loc)/history/history.h <(src_loc)/history/history_admin_log_filter.cpp @@ -546,8 +550,6 @@ <(src_loc)/autoupdater.h <(src_loc)/config.h <(src_loc)/countries.h -<(src_loc)/dialogswidget.cpp -<(src_loc)/dialogswidget.h <(src_loc)/facades.cpp <(src_loc)/facades.h <(src_loc)/layerwidget.cpp