476 lines
14 KiB
C++
476 lines
14 KiB
C++
/*
|
|
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"
|
|
|
|
namespace Dialogs {
|
|
class Row;
|
|
class FakeRow;
|
|
class IndexedList;
|
|
} // namespace Dialogs
|
|
|
|
namespace Ui {
|
|
class IconButton;
|
|
class PopupMenu;
|
|
class DropdownMenu;
|
|
class FlatButton;
|
|
class LinkButton;
|
|
class FlatInput;
|
|
class CrossButton;
|
|
template <typename Widget>
|
|
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
|
|
|
|
public:
|
|
DialogsInner(QWidget *parent, gsl::not_null<Window::Controller*> controller, QWidget *main);
|
|
|
|
void dialogsReceived(const QVector<MTPDialog> &dialogs);
|
|
void addSavedPeersAfter(const QDateTime &date);
|
|
void addAllSavedPeers();
|
|
bool searchReceived(const QVector<MTPMessage> &result, DialogsSearchRequestType type, int32 fullCount);
|
|
void peerSearchReceived(const QString &query, const QVector<MTPPeer> &result);
|
|
void showMore(int32 pixels);
|
|
|
|
void activate();
|
|
|
|
void contactsReceived(const QVector<MTPContact> &result);
|
|
|
|
void selectSkip(int32 direction);
|
|
void selectSkipPage(int32 pixels, int32 direction);
|
|
|
|
void createDialog(History *history);
|
|
void dlgUpdated(Dialogs::Mode list, Dialogs::Row *row);
|
|
void dlgUpdated(PeerData *peer, MsgId msgId);
|
|
void removeDialog(History *history);
|
|
|
|
void dragLeft();
|
|
|
|
void clearFilter();
|
|
void refresh(bool toTop = false);
|
|
|
|
bool choosePeer();
|
|
void saveRecentHashtags(const QString &text);
|
|
|
|
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);
|
|
|
|
Dialogs::IndexedList *contactsList();
|
|
Dialogs::IndexedList *dialogsList();
|
|
Dialogs::IndexedList *contactsNoDialogsList();
|
|
int32 lastSearchDate() const;
|
|
PeerData *lastSearchPeer() const;
|
|
MsgId lastSearchId() const;
|
|
MsgId lastSearchMigratedId() const;
|
|
|
|
void setMouseSelection(bool mouseSelection, bool toTop = false);
|
|
|
|
enum State {
|
|
DefaultState = 0,
|
|
FilteredState = 1,
|
|
SearchedState = 2,
|
|
};
|
|
void setState(State newState);
|
|
State state() const;
|
|
bool hasFilteredResults() const;
|
|
|
|
void searchInPeer(PeerData *peer);
|
|
|
|
void onFilterUpdate(QString newFilter, bool force = false);
|
|
void onHashtagFilterUpdate(QStringRef newFilter);
|
|
|
|
PeerData *updateFromParentDrag(QPoint globalPos);
|
|
|
|
void setLoadMoreCallback(base::lambda<void()> callback) {
|
|
_loadMoreCallback = std::move(callback);
|
|
}
|
|
void setVisibleTopBottom(int visibleTop, int visibleBottom) override;
|
|
|
|
void notify_userIsContactChanged(UserData *user, bool fromThisApp);
|
|
void notify_historyMuteUpdated(History *history);
|
|
|
|
~DialogsInner();
|
|
|
|
public slots:
|
|
void onParentGeometryChanged();
|
|
void onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars);
|
|
void onPeerPhotoChanged(PeerData *peer);
|
|
void onDialogRowReplaced(Dialogs::Row *oldRow, Dialogs::Row *newRow);
|
|
|
|
void onMenuDestroyed(QObject*);
|
|
|
|
signals:
|
|
void draggingScrollDelta(int delta);
|
|
void mustScrollTo(int scrollToTop, int scrollToBottom);
|
|
void dialogMoved(int movedFrom, int movedTo);
|
|
void searchMessages();
|
|
void searchResultChosen();
|
|
void cancelSearchInPeer();
|
|
void completeHashtag(QString tag);
|
|
void refreshHashtags();
|
|
|
|
protected:
|
|
void paintRegion(Painter &p, const QRegion ®ion, bool paintingOther) override;
|
|
void mouseMoveEvent(QMouseEvent *e) override;
|
|
void mousePressEvent(QMouseEvent *e) override;
|
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
|
void resizeEvent(QResizeEvent *e) override;
|
|
void enterEventHook(QEvent *e) override;
|
|
void leaveEventHook(QEvent *e) override;
|
|
void contextMenuEvent(QContextMenuEvent *e) override;
|
|
|
|
private:
|
|
struct ImportantSwitch;
|
|
using DialogsList = std::unique_ptr<Dialogs::IndexedList>;
|
|
using FilteredDialogs = QVector<Dialogs::Row*>;
|
|
using SearchResults = std::vector<std::unique_ptr<Dialogs::FakeRow>>;
|
|
struct HashtagResult;
|
|
using HashtagResults = std::vector<std::unique_ptr<HashtagResult>>;
|
|
struct PeerSearchResult;
|
|
using PeerSearchResults = std::vector<std::unique_ptr<PeerSearchResult>>;
|
|
|
|
void mousePressReleased(Qt::MouseButton button);
|
|
void clearIrrelevantState();
|
|
void updateSelected() {
|
|
updateSelected(mapFromGlobal(QCursor::pos()));
|
|
}
|
|
void updateSelected(QPoint localPos);
|
|
void loadPeerPhotos();
|
|
void setImportantSwitchPressed(bool pressed);
|
|
void setPressed(Dialogs::Row *pressed);
|
|
void setHashtagPressed(int pressed);
|
|
void setFilteredPressed(int pressed);
|
|
void setPeerSearchPressed(int pressed);
|
|
void setSearchedPressed(int pressed);
|
|
bool isPressed() const {
|
|
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);
|
|
}
|
|
|
|
void itemRemoved(HistoryItem *item);
|
|
enum class UpdateRowSection {
|
|
Default = 0x01,
|
|
Filtered = 0x02,
|
|
PeerSearch = 0x04,
|
|
MessageSearch = 0x08,
|
|
All = 0x0F,
|
|
};
|
|
Q_DECLARE_FLAGS(UpdateRowSections, UpdateRowSection);
|
|
Q_DECLARE_FRIEND_OPERATORS_FOR_FLAGS(UpdateRowSections);
|
|
void updateDialogRow(PeerData *peer, MsgId msgId, QRect updateRect, UpdateRowSections sections = UpdateRowSection::All);
|
|
|
|
int dialogsOffset() const;
|
|
int filteredOffset() const;
|
|
int peerSearchOffset() const;
|
|
int searchedOffset() const;
|
|
|
|
void paintDialog(Painter &p, Dialogs::Row *row, int fullWidth, PeerData *active, PeerData *selected, bool onlyBackground, TimeMs ms);
|
|
void paintPeerSearchResult(Painter &p, const PeerSearchResult *result, int fullWidth, bool active, bool selected, bool onlyBackground, TimeMs ms) const;
|
|
void paintSearchInPeer(Painter &p, int fullWidth, bool onlyBackground) const;
|
|
|
|
void clearSelection();
|
|
void clearSearchResults(bool clearPeerSearchResults = true);
|
|
void updateSelectedRow(PeerData *peer = 0);
|
|
|
|
Dialogs::IndexedList *shownDialogs() const {
|
|
return (Global::DialogsMode() == Dialogs::Mode::Important) ? _dialogsImportant.get() : _dialogs.get();
|
|
}
|
|
|
|
void checkReorderPinnedStart(QPoint localPosition);
|
|
int shownPinnedCount() const;
|
|
int updateReorderIndexGetCount();
|
|
bool updateReorderPinned(QPoint localPosition);
|
|
void finishReorderPinned();
|
|
void stopReorderPinned();
|
|
int countPinnedIndex(Dialogs::Row *ofRow);
|
|
void savePinnedOrder();
|
|
void step_pinnedShifting(TimeMs ms, bool timer);
|
|
|
|
gsl::not_null<Window::Controller*> _controller;
|
|
|
|
DialogsList _dialogs;
|
|
DialogsList _dialogsImportant;
|
|
|
|
DialogsList _contactsNoDialogs;
|
|
DialogsList _contacts;
|
|
|
|
bool _mouseSelection = false;
|
|
Qt::MouseButton _pressButton = Qt::LeftButton;
|
|
|
|
std::unique_ptr<ImportantSwitch> _importantSwitch;
|
|
bool _importantSwitchSelected = false;
|
|
bool _importantSwitchPressed = false;
|
|
Dialogs::Row *_selected = nullptr;
|
|
Dialogs::Row *_pressed = nullptr;
|
|
|
|
Dialogs::Row *_dragging = nullptr;
|
|
int _draggingIndex = -1;
|
|
int _aboveIndex = -1;
|
|
QPoint _dragStart;
|
|
struct PinnedRow {
|
|
anim::value yadd;
|
|
TimeMs animStartTime = 0;
|
|
};
|
|
std::vector<PinnedRow> _pinnedRows;
|
|
BasicAnimation _a_pinnedShifting;
|
|
QList<History*> _pinnedOrder;
|
|
|
|
// Remember the last currently dragged row top shift for updating area.
|
|
int _aboveTopShift = -1;
|
|
|
|
int _visibleTop = 0;
|
|
int _visibleBottom = 0;
|
|
QString _filter, _hashtagFilter;
|
|
|
|
HashtagResults _hashtagResults;
|
|
int _hashtagSelected = -1;
|
|
int _hashtagPressed = -1;
|
|
bool _hashtagDeleteSelected = false;
|
|
bool _hashtagDeletePressed = false;
|
|
|
|
FilteredDialogs _filterResults;
|
|
int _filteredSelected = -1;
|
|
int _filteredPressed = -1;
|
|
|
|
QString _peerSearchQuery;
|
|
PeerSearchResults _peerSearchResults;
|
|
int _peerSearchSelected = -1;
|
|
int _peerSearchPressed = -1;
|
|
|
|
SearchResults _searchResults;
|
|
int _searchedCount = 0;
|
|
int _searchedMigratedCount = 0;
|
|
int _searchedSelected = -1;
|
|
int _searchedPressed = -1;
|
|
|
|
int _lastSearchDate = 0;
|
|
PeerData *_lastSearchPeer = nullptr;
|
|
MsgId _lastSearchId = 0;
|
|
MsgId _lastSearchMigratedId = 0;
|
|
|
|
State _state = DefaultState;
|
|
|
|
object_ptr<Ui::LinkButton> _addContactLnk;
|
|
object_ptr<Ui::IconButton> _cancelSearchInPeer;
|
|
|
|
PeerData *_searchInPeer = nullptr;
|
|
PeerData *_searchInMigrated = nullptr;
|
|
PeerData *_menuPeer = nullptr;
|
|
|
|
Ui::PopupMenu *_menu = nullptr;
|
|
|
|
base::lambda<void()> _loadMoreCallback;
|
|
|
|
};
|
|
|
|
Q_DECLARE_OPERATORS_FOR_FLAGS(DialogsInner::UpdateRowSections);
|
|
|
|
class DialogsWidget : public Window::AbstractSectionWidget, public RPCSender {
|
|
Q_OBJECT
|
|
|
|
public:
|
|
DialogsWidget(QWidget *parent, gsl::not_null<Window::Controller*> 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 showMainMenu();
|
|
void updateLockUnlockVisibility();
|
|
void updateJumpToDateVisibility(bool fast = false);
|
|
void updateControlsGeometry();
|
|
void updateForwardBar();
|
|
|
|
void unreadCountsReceived(const QVector<MTPDialog> &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<Ui::IconButton> _forwardCancel = { nullptr };
|
|
object_ptr<Ui::IconButton> _mainMenuToggle;
|
|
object_ptr<Ui::FlatInput> _filter;
|
|
object_ptr<Ui::WidgetScaledFadeWrap<Ui::IconButton>> _jumpToDate;
|
|
object_ptr<Ui::CrossButton> _cancelSearch;
|
|
object_ptr<Ui::IconButton> _lockUnlock;
|
|
object_ptr<Ui::ScrollArea> _scroll;
|
|
QPointer<DialogsInner> _inner;
|
|
class UpdateButton;
|
|
object_ptr<UpdateButton> _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<QString, MTPmessages_Messages>;
|
|
SearchCache _searchCache;
|
|
|
|
using SearchQueries = QMap<mtpRequestId, QString>;
|
|
SearchQueries _searchQueries;
|
|
|
|
using PeerSearchCache = QMap<QString, MTPcontacts_Found>;
|
|
PeerSearchCache _peerSearchCache;
|
|
|
|
using PeerSearchQueries = QMap<mtpRequestId, QString>;
|
|
PeerSearchQueries _peerSearchQueries;
|
|
|
|
QPixmap _widthAnimationCache;
|
|
|
|
object_ptr<QTimer> _draggingScrollTimer = { nullptr };
|
|
int _draggingScrollDelta = 0;
|
|
|
|
};
|