468 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			468 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
 | |
| 
 | |
| enum DialogsSearchRequestType {
 | |
| 	DialogsSearchFromStart,
 | |
| 	DialogsSearchFromOffset,
 | |
| 	DialogsSearchPeerFromStart,
 | |
| 	DialogsSearchPeerFromOffset,
 | |
| 	DialogsSearchMigratedFromStart,
 | |
| 	DialogsSearchMigratedFromOffset,
 | |
| };
 | |
| 
 | |
| class DialogsInner : public Ui::SplittedWidget, public RPCSender, private base::Subscriber {
 | |
| 	Q_OBJECT
 | |
| 
 | |
| public:
 | |
| 	DialogsInner(QWidget *parent, 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();
 | |
| 	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);
 | |
| 
 | |
| 	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 TWidget, public RPCSender, private base::Subscriber {
 | |
| 	Q_OBJECT
 | |
| 
 | |
| public:
 | |
| 	DialogsWidget(QWidget *parent);
 | |
| 
 | |
| 	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();
 | |
| 
 | |
| 	void searchMessages(const QString &query, PeerData *inPeer = 0);
 | |
| 	void onSearchMore();
 | |
| 
 | |
| 	void rpcClear() override {
 | |
| 		_inner->rpcClear();
 | |
| 		RPCSender::rpcClear();
 | |
| 	}
 | |
| 
 | |
| 	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();
 | |
| 
 | |
| 	bool _dragInScroll = false;
 | |
| 	bool _dragForward = false;
 | |
| 	QTimer _chooseByDragTimer;
 | |
| 
 | |
| 	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 _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;
 | |
| 
 | |
| };
 | 
